2012-01-04 20:54:20 +04:00
# include "../perf.h"
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
# include "util.h"
2014-07-15 01:46:48 +04:00
# include "debug.h"
2013-12-09 20:14:24 +04:00
# include <api/fs/fs.h>
2010-01-16 16:21:15 +03:00
# include <sys/mman.h>
2015-11-06 16:55:35 +03:00
# include <sys/utsname.h>
2013-09-30 14:07:11 +04:00
# ifdef HAVE_BACKTRACE_SUPPORT
2012-08-08 06:32:05 +04:00
# include <execinfo.h>
2012-09-08 04:43:23 +04:00
# endif
2012-08-08 06:32:05 +04:00
# include <stdio.h>
# include <stdlib.h>
2013-12-03 17:09:22 +04:00
# include <string.h>
# include <errno.h>
2013-12-11 16:36:23 +04:00
# include <limits.h>
2013-12-11 16:36:32 +04:00
# include <byteswap.h>
2013-11-28 14:30:15 +04:00
# include <linux/kernel.h>
2016-02-24 14:20:44 +03:00
# include <linux/log2.h>
2016-08-05 21:40:30 +03:00
# include <linux/time64.h>
2014-08-11 12:50:02 +04:00
# include <unistd.h>
2014-10-01 20:00:26 +04:00
# include "callchain.h"
2016-01-07 14:41:53 +03:00
# include "strlist.h"
2014-10-01 20:00:26 +04:00
2016-07-04 15:16:21 +03:00
# define CALLCHAIN_PARAM_DEFAULT \
. mode = CHAIN_GRAPH_ABS , \
. min_percent = 0.5 , \
. order = ORDER_CALLEE , \
. key = CCKEY_FUNCTION , \
. value = CCVAL_PERCENT , \
struct callchain_param callchain_param = {
CALLCHAIN_PARAM_DEFAULT
} ;
struct callchain_param callchain_param_default = {
CALLCHAIN_PARAM_DEFAULT
2014-10-01 20:00:26 +04:00
} ;
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
2012-01-04 20:54:20 +04:00
/*
* XXX We need to find a better place for these things . . .
*/
2012-10-06 21:57:10 +04:00
unsigned int page_size ;
2014-05-31 00:10:05 +04:00
int cacheline_size ;
2012-10-06 21:57:10 +04:00
2016-05-17 03:16:54 +03:00
int sysctl_perf_event_max_stack = PERF_MAX_STACK_DEPTH ;
int sysctl_perf_event_max_contexts_per_stack = PERF_MAX_CONTEXTS_PER_STACK ;
2016-04-27 16:16:24 +03:00
2012-12-13 23:43:04 +04:00
bool test_attr__enabled ;
2012-01-04 20:54:20 +04:00
bool perf_host = true ;
2012-02-10 21:05:05 +04:00
bool perf_guest = false ;
2012-01-04 20:54:20 +04:00
void event_attr_init ( struct perf_event_attr * attr )
{
if ( ! perf_host )
attr - > exclude_host = 1 ;
if ( ! perf_guest )
attr - > exclude_guest = 1 ;
2012-02-09 19:12:38 +04:00
/* to capture ABI version */
attr - > size = sizeof ( * attr ) ;
2012-01-04 20:54:20 +04:00
}
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
int mkdir_p ( char * path , mode_t mode )
{
struct stat st ;
int err ;
char * d = path ;
if ( * d ! = ' / ' )
return - 1 ;
if ( stat ( path , & st ) = = 0 )
return 0 ;
while ( * + + d = = ' / ' ) ;
while ( ( d = strchr ( d , ' / ' ) ) ) {
* d = ' \0 ' ;
err = stat ( path , & st ) & & mkdir ( path , mode ) ;
* d + + = ' / ' ;
if ( err )
return - 1 ;
while ( * d = = ' / ' )
+ + d ;
}
return ( stat ( path , & st ) & & mkdir ( path , mode ) ) ? - 1 : 0 ;
}
2017-01-27 00:19:59 +03:00
int rm_rf ( const char * path )
2015-05-18 03:30:17 +03:00
{
DIR * dir ;
int ret = 0 ;
struct dirent * d ;
char namebuf [ PATH_MAX ] ;
dir = opendir ( path ) ;
if ( dir = = NULL )
return 0 ;
while ( ( d = readdir ( dir ) ) ! = NULL & & ! ret ) {
struct stat statbuf ;
if ( ! strcmp ( d - > d_name , " . " ) | | ! strcmp ( d - > d_name , " .. " ) )
continue ;
scnprintf ( namebuf , sizeof ( namebuf ) , " %s/%s " ,
path , d - > d_name ) ;
2016-06-08 12:29:11 +03:00
/* We have to check symbolic link itself */
ret = lstat ( namebuf , & statbuf ) ;
2015-05-18 03:30:17 +03:00
if ( ret < 0 ) {
pr_debug ( " stat failed: %s \n " , namebuf ) ;
break ;
}
2016-06-08 12:29:11 +03:00
if ( S_ISDIR ( statbuf . st_mode ) )
2015-05-18 03:30:17 +03:00
ret = rm_rf ( namebuf ) ;
2016-06-08 12:29:11 +03:00
else
ret = unlink ( namebuf ) ;
2015-05-18 03:30:17 +03:00
}
closedir ( dir ) ;
if ( ret < 0 )
return ret ;
return rmdir ( path ) ;
}
2016-04-26 12:02:42 +03:00
/* A filter which removes dot files */
bool lsdir_no_dot_filter ( const char * name __maybe_unused , struct dirent * d )
{
return d - > d_name [ 0 ] ! = ' . ' ;
}
/* lsdir reads a directory and store it in strlist */
struct strlist * lsdir ( const char * name ,
bool ( * filter ) ( const char * , struct dirent * ) )
{
struct strlist * list = NULL ;
DIR * dir ;
struct dirent * d ;
dir = opendir ( name ) ;
if ( ! dir )
return NULL ;
list = strlist__new ( NULL , NULL ) ;
if ( ! list ) {
2016-05-11 16:51:27 +03:00
errno = ENOMEM ;
2016-04-26 12:02:42 +03:00
goto out ;
}
while ( ( d = readdir ( dir ) ) ! = NULL ) {
if ( ! filter | | filter ( name , d ) )
strlist__add ( list , d - > d_name ) ;
}
out :
closedir ( dir ) ;
return list ;
}
2015-06-08 17:50:16 +03:00
static int slow_copyfile ( const char * from , const char * to )
2010-01-14 23:30:06 +03:00
{
2013-10-14 14:43:41 +04:00
int err = - 1 ;
2010-01-14 23:30:06 +03:00
char * line = NULL ;
size_t n ;
FILE * from_fp = fopen ( from , " r " ) , * to_fp ;
if ( from_fp = = NULL )
goto out ;
to_fp = fopen ( to , " w " ) ;
if ( to_fp = = NULL )
goto out_fclose_from ;
while ( getline ( & line , & n , from_fp ) > 0 )
if ( fputs ( line , to_fp ) = = EOF )
goto out_fclose_to ;
err = 0 ;
out_fclose_to :
fclose ( to_fp ) ;
free ( line ) ;
out_fclose_from :
fclose ( from_fp ) ;
out :
return err ;
}
2015-05-18 03:30:18 +03:00
int copyfile_offset ( int ifd , loff_t off_in , int ofd , loff_t off_out , u64 size )
{
void * ptr ;
loff_t pgoff ;
pgoff = off_in & ~ ( page_size - 1 ) ;
off_in - = pgoff ;
ptr = mmap ( NULL , off_in + size , PROT_READ , MAP_PRIVATE , ifd , pgoff ) ;
if ( ptr = = MAP_FAILED )
return - 1 ;
while ( size ) {
ssize_t ret = pwrite ( ofd , ptr + off_in , size , off_out ) ;
if ( ret < 0 & & errno = = EINTR )
continue ;
if ( ret < = 0 )
break ;
size - = ret ;
off_in + = ret ;
off_out - = ret ;
}
munmap ( ptr , off_in + size ) ;
return size ? - 1 : 0 ;
}
2013-10-14 14:43:41 +04:00
int copyfile_mode ( const char * from , const char * to , mode_t mode )
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
{
int fromfd , tofd ;
struct stat st ;
int err = - 1 ;
2015-06-08 17:50:16 +03:00
char * tmp = NULL , * ptr = NULL ;
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
if ( stat ( from , & st ) )
goto out ;
2015-06-08 17:50:16 +03:00
/* extra 'x' at the end is to reserve space for '.' */
if ( asprintf ( & tmp , " %s.XXXXXXx " , to ) < 0 ) {
tmp = NULL ;
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
goto out ;
2015-06-08 17:50:16 +03:00
}
ptr = strrchr ( tmp , ' / ' ) ;
if ( ! ptr )
goto out ;
ptr = memmove ( ptr + 1 , ptr , strlen ( ptr ) - 1 ) ;
* ptr = ' . ' ;
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
2015-06-08 17:50:16 +03:00
tofd = mkstemp ( tmp ) ;
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
if ( tofd < 0 )
2015-06-08 17:50:16 +03:00
goto out ;
if ( fchmod ( tofd , mode ) )
goto out_close_to ;
if ( st . st_size = = 0 ) { /* /proc? do it slowly... */
err = slow_copyfile ( from , tmp ) ;
goto out_close_to ;
}
fromfd = open ( from , O_RDONLY ) ;
if ( fromfd < 0 )
goto out_close_to ;
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
2015-05-18 03:30:18 +03:00
err = copyfile_offset ( fromfd , 0 , tofd , 0 , st . st_size ) ;
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
close ( fromfd ) ;
2015-06-08 17:50:16 +03:00
out_close_to :
close ( tofd ) ;
if ( ! err )
err = link ( tmp , to ) ;
unlink ( tmp ) ;
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
out :
2015-06-08 17:50:16 +03:00
free ( tmp ) ;
perf record: Introduce a symtab cache
Now a cache will be created in a ~/.debug debuginfo like
hierarchy, so that at the end of a 'perf record' session all the
binaries (with build-ids) involved get collected and indexed by
their build-ids, so that perf report can find them.
This is interesting when developing software where you want to
do a 'perf diff' with the previous build and opens avenues for
lots more interesting tools, like a 'perf diff --graph' that
takes more than two binaries into account.
Tunables for collecting just the symtabs can be added if one
doesn't want to have the full binary, but having the full binary
allows things like 'perf rerecord' or other tools that can
re-run the tests by having access to the exact binary in some
perf.data file, so it may well be interesting to keep the full
binary there.
Space consumption is minimised by trying to use hard links, a
'perf cache' tool to manage the space used, a la ccache is
required to purge older entries.
With this in place it will be possible also to introduce new
commands, 'perf archive' and 'perf restore' (or some more
suitable and future proof names) to create a cpio/tar file with
the perf data and the files in the cache that _had_ perf hits of
interest.
There are more aspects to polish, like finding the right vmlinux
file to cache, etc, but this is enough for a first step.
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1261957026-15580-10-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-12-28 02:37:06 +03:00
return err ;
}
2010-05-14 21:19:35 +04:00
2013-10-14 14:43:41 +04:00
int copyfile ( const char * from , const char * to )
{
return copyfile_mode ( from , to , 0755 ) ;
}
2010-05-14 21:19:35 +04:00
unsigned long convert_unit ( unsigned long value , char * unit )
{
* unit = ' ' ;
if ( value > 1000 ) {
value / = 1000 ;
* unit = ' K ' ;
}
if ( value > 1000 ) {
value / = 1000 ;
* unit = ' M ' ;
}
if ( value > 1000 ) {
value / = 1000 ;
* unit = ' G ' ;
}
return value ;
}
2011-01-03 21:50:55 +03:00
2013-11-28 14:30:16 +04:00
static ssize_t ion ( bool is_read , int fd , void * buf , size_t n )
2011-01-03 21:50:55 +03:00
{
void * buf_start = buf ;
2013-11-28 14:30:15 +04:00
size_t left = n ;
2011-01-03 21:50:55 +03:00
2013-11-28 14:30:15 +04:00
while ( left ) {
2013-11-28 14:30:16 +04:00
ssize_t ret = is_read ? read ( fd , buf , left ) :
write ( fd , buf , left ) ;
2011-01-03 21:50:55 +03:00
2014-04-24 17:27:32 +04:00
if ( ret < 0 & & errno = = EINTR )
continue ;
2011-01-03 21:50:55 +03:00
if ( ret < = 0 )
return ret ;
2013-11-28 14:30:15 +04:00
left - = ret ;
buf + = ret ;
2011-01-03 21:50:55 +03:00
}
2013-11-28 14:30:15 +04:00
BUG_ON ( ( size_t ) ( buf - buf_start ) ! = n ) ;
return n ;
2011-01-03 21:50:55 +03:00
}
2012-04-19 20:15:24 +04:00
2013-11-28 14:30:16 +04:00
/*
* Read exactly ' n ' bytes or return an error .
*/
ssize_t readn ( int fd , void * buf , size_t n )
{
return ion ( true , fd , buf , n ) ;
}
/*
* Write exactly ' n ' bytes or return an error .
*/
ssize_t writen ( int fd , void * buf , size_t n )
{
return ion ( false , fd , buf , n ) ;
}
2012-04-19 20:15:24 +04:00
size_t hex_width ( u64 v )
{
size_t n = 1 ;
while ( ( v > > = 4 ) )
+ + n ;
return n ;
}
2012-08-08 06:32:05 +04:00
2012-10-28 01:18:30 +04:00
static int hex ( char ch )
{
if ( ( ch > = ' 0 ' ) & & ( ch < = ' 9 ' ) )
return ch - ' 0 ' ;
if ( ( ch > = ' a ' ) & & ( ch < = ' f ' ) )
return ch - ' a ' + 10 ;
if ( ( ch > = ' A ' ) & & ( ch < = ' F ' ) )
return ch - ' A ' + 10 ;
return - 1 ;
}
/*
* While we find nice hex chars , build a long_val .
* Return number of chars processed .
*/
int hex2u64 ( const char * ptr , u64 * long_val )
{
const char * p = ptr ;
* long_val = 0 ;
while ( * p ) {
const int hex_val = hex ( * p ) ;
if ( hex_val < 0 )
break ;
* long_val = ( * long_val < < 4 ) | hex_val ;
p + + ;
}
return p - ptr ;
}
2012-08-08 06:32:05 +04:00
/* Obtain a backtrace and print it to stdout. */
2013-09-30 14:07:11 +04:00
# ifdef HAVE_BACKTRACE_SUPPORT
2012-08-08 06:32:05 +04:00
void dump_stack ( void )
{
void * array [ 16 ] ;
size_t size = backtrace ( array , ARRAY_SIZE ( array ) ) ;
char * * strings = backtrace_symbols ( array , size ) ;
size_t i ;
printf ( " Obtained %zd stack frames. \n " , size ) ;
for ( i = 0 ; i < size ; i + + )
printf ( " %s \n " , strings [ i ] ) ;
free ( strings ) ;
}
2012-09-08 04:43:23 +04:00
# else
void dump_stack ( void ) { }
# endif
2013-01-14 21:48:01 +04:00
2015-02-24 21:34:23 +03:00
void sighandler_dump_stack ( int sig )
{
psignal ( sig , " perf " ) ;
dump_stack ( ) ;
2015-12-11 22:43:57 +03:00
signal ( sig , SIG_DFL ) ;
raise ( sig ) ;
2015-02-24 21:34:23 +03:00
}
perf tools: Introduce timestamp__scnprintf_usec()
Joonwoo reported that there's a mismatch between timestamps in script
and sched commands. This was because of difference in printing the
timestamp. Factor out the code and share it so that they can be in
sync. Also I found that sched map has similar problem, fix it too.
Committer notes:
Fixed the max_lat_at bug introduced by Namhyung's original patch, as
pointed out by Joonwoo, and made it a function following the scnprintf()
model, i.e. returning the number of bytes formatted, and receiving as
the first parameter the object from where the data to the formatting is
obtained, renaming it from:
char *timestamp_in_usec(char *bf, size_t size, u64 timestamp)
to
int timestamp__scnprintf_usec(u64 timestamp, char *bf, size_t size)
Reported-by: Joonwoo Park <joonwoop@codeaurora.org>
Signed-off-by: Namhyung Kim <namhyung@kernel.org>
Cc: David Ahern <dsahern@gmail.com>
Cc: Jiri Olsa <jolsa@kernel.org>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Link: http://lkml.kernel.org/r/20161024020246.14928-3-namhyung@kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
2016-10-24 05:02:45 +03:00
int timestamp__scnprintf_usec ( u64 timestamp , char * buf , size_t sz )
{
u64 sec = timestamp / NSEC_PER_SEC ;
u64 usec = ( timestamp % NSEC_PER_SEC ) / NSEC_PER_USEC ;
return scnprintf ( buf , sz , " % " PRIu64 " .%06 " PRIu64 , sec , usec ) ;
}
2013-09-01 14:36:13 +04:00
unsigned long parse_tag_value ( const char * str , struct parse_tag * tags )
{
struct parse_tag * i = tags ;
while ( i - > tag ) {
char * s ;
s = strchr ( str , i - > tag ) ;
if ( s ) {
unsigned long int value ;
char * endptr ;
value = strtoul ( str , & endptr , 10 ) ;
if ( s ! = endptr )
break ;
2013-10-22 11:34:17 +04:00
if ( value > ULONG_MAX / i - > mult )
break ;
2013-09-01 14:36:13 +04:00
value * = i - > mult ;
return value ;
}
i + + ;
}
return ( unsigned long ) - 1 ;
}
2013-10-17 23:33:43 +04:00
2015-08-06 22:44:52 +03:00
int get_stack_size ( const char * str , unsigned long * _size )
{
char * endptr ;
unsigned long size ;
unsigned long max_size = round_down ( USHRT_MAX , sizeof ( u64 ) ) ;
size = strtoul ( str , & endptr , 0 ) ;
do {
if ( * endptr )
break ;
size = round_up ( size , sizeof ( u64 ) ) ;
if ( ! size | | size > max_size )
break ;
* _size = size ;
return 0 ;
} while ( 0 ) ;
pr_err ( " callchain: Incorrect stack dump size (max %ld): %s \n " ,
max_size , str ) ;
return - 1 ;
}
int parse_callchain_record ( const char * arg , struct callchain_param * param )
{
char * tok , * name , * saveptr = NULL ;
char * buf ;
int ret = - 1 ;
/* We need buffer that we know we can write to. */
buf = malloc ( strlen ( arg ) + 1 ) ;
if ( ! buf )
return - ENOMEM ;
strcpy ( buf , arg ) ;
tok = strtok_r ( ( char * ) buf , " , " , & saveptr ) ;
name = tok ? : ( char * ) buf ;
do {
/* Framepointer style */
if ( ! strncmp ( name , " fp " , sizeof ( " fp " ) ) ) {
if ( ! strtok_r ( NULL , " , " , & saveptr ) ) {
param - > record_mode = CALLCHAIN_FP ;
ret = 0 ;
} else
pr_err ( " callchain: No more arguments "
" needed for --call-graph fp \n " ) ;
break ;
/* Dwarf style */
} else if ( ! strncmp ( name , " dwarf " , sizeof ( " dwarf " ) ) ) {
const unsigned long default_stack_dump_size = 8192 ;
ret = 0 ;
param - > record_mode = CALLCHAIN_DWARF ;
param - > dump_size = default_stack_dump_size ;
tok = strtok_r ( NULL , " , " , & saveptr ) ;
if ( tok ) {
unsigned long size = 0 ;
ret = get_stack_size ( tok , & size ) ;
param - > dump_size = size ;
}
} else if ( ! strncmp ( name , " lbr " , sizeof ( " lbr " ) ) ) {
if ( ! strtok_r ( NULL , " , " , & saveptr ) ) {
param - > record_mode = CALLCHAIN_LBR ;
ret = 0 ;
} else
pr_err ( " callchain: No more arguments "
" needed for --call-graph lbr \n " ) ;
break ;
} else {
pr_err ( " callchain: Unknown --call-graph option "
" value: %s \n " , arg ) ;
break ;
}
} while ( 0 ) ;
free ( buf ) ;
return ret ;
}
2013-12-07 02:25:51 +04:00
const char * get_filename_for_perf_kvm ( void )
{
const char * filename ;
if ( perf_host & & ! perf_guest )
filename = strdup ( " perf.data.host " ) ;
else if ( ! perf_host & & perf_guest )
filename = strdup ( " perf.data.guest " ) ;
else
filename = strdup ( " perf.data.kvm " ) ;
return filename ;
}
2013-12-11 16:36:23 +04:00
int perf_event_paranoid ( void )
{
int value ;
2014-12-11 19:37:59 +03:00
if ( sysctl__read_int ( " kernel/perf_event_paranoid " , & value ) )
2013-12-11 16:36:23 +04:00
return INT_MAX ;
return value ;
}
2013-12-11 16:36:32 +04:00
void mem_bswap_32 ( void * src , int byte_size )
{
u32 * m = src ;
while ( byte_size > 0 ) {
* m = bswap_32 ( * m ) ;
byte_size - = sizeof ( u32 ) ;
+ + m ;
}
}
void mem_bswap_64 ( void * src , int byte_size )
{
u64 * m = src ;
while ( byte_size > 0 ) {
* m = bswap_64 ( * m ) ;
byte_size - = sizeof ( u64 ) ;
+ + m ;
}
}
2014-08-01 19:46:54 +04:00
bool find_process ( const char * name )
{
size_t len = strlen ( name ) ;
DIR * dir ;
struct dirent * d ;
int ret = - 1 ;
dir = opendir ( procfs__mountpoint ( ) ) ;
if ( ! dir )
2015-09-17 13:08:53 +03:00
return false ;
2014-08-01 19:46:54 +04:00
/* Walk through the directory. */
while ( ret & & ( d = readdir ( dir ) ) ! = NULL ) {
char path [ PATH_MAX ] ;
char * data ;
size_t size ;
if ( ( d - > d_type ! = DT_DIR ) | |
! strcmp ( " . " , d - > d_name ) | |
! strcmp ( " .. " , d - > d_name ) )
continue ;
scnprintf ( path , sizeof ( path ) , " %s/%s/comm " ,
procfs__mountpoint ( ) , d - > d_name ) ;
if ( filename__read_str ( path , & data , & size ) )
continue ;
ret = strncmp ( name , data , len ) ;
free ( data ) ;
}
closedir ( dir ) ;
return ret ? false : true ;
}
2015-11-06 16:55:35 +03:00
2016-11-15 07:05:44 +03:00
static int
fetch_ubuntu_kernel_version ( unsigned int * puint )
{
ssize_t len ;
size_t line_len = 0 ;
char * ptr , * line = NULL ;
int version , patchlevel , sublevel , err ;
FILE * vsig = fopen ( " /proc/version_signature " , " r " ) ;
if ( ! vsig ) {
pr_debug ( " Open /proc/version_signature failed: %s \n " ,
strerror ( errno ) ) ;
return - 1 ;
}
len = getline ( & line , & line_len , vsig ) ;
fclose ( vsig ) ;
err = - 1 ;
if ( len < = 0 ) {
pr_debug ( " Reading from /proc/version_signature failed: %s \n " ,
strerror ( errno ) ) ;
goto errout ;
}
ptr = strrchr ( line , ' ' ) ;
if ( ! ptr ) {
pr_debug ( " Parsing /proc/version_signature failed: %s \n " , line ) ;
goto errout ;
}
err = sscanf ( ptr + 1 , " %d.%d.%d " ,
& version , & patchlevel , & sublevel ) ;
if ( err ! = 3 ) {
pr_debug ( " Unable to get kernel version from /proc/version_signature '%s' \n " ,
line ) ;
goto errout ;
}
if ( puint )
* puint = ( version < < 16 ) + ( patchlevel < < 8 ) + sublevel ;
err = 0 ;
errout :
free ( line ) ;
return err ;
}
2015-11-06 16:55:35 +03:00
int
fetch_kernel_version ( unsigned int * puint , char * str ,
size_t str_size )
{
struct utsname utsname ;
int version , patchlevel , sublevel , err ;
2016-11-15 07:05:44 +03:00
bool int_ver_ready = false ;
if ( access ( " /proc/version_signature " , R_OK ) = = 0 )
if ( ! fetch_ubuntu_kernel_version ( puint ) )
int_ver_ready = true ;
2015-11-06 16:55:35 +03:00
if ( uname ( & utsname ) )
return - 1 ;
if ( str & & str_size ) {
strncpy ( str , utsname . release , str_size ) ;
str [ str_size - 1 ] = ' \0 ' ;
}
err = sscanf ( utsname . release , " %d.%d.%d " ,
& version , & patchlevel , & sublevel ) ;
if ( err ! = 3 ) {
2016-11-15 07:05:44 +03:00
pr_debug ( " Unable to get kernel version from uname '%s' \n " ,
2015-11-06 16:55:35 +03:00
utsname . release ) ;
return - 1 ;
}
2016-11-15 07:05:44 +03:00
if ( puint & & ! int_ver_ready )
2015-11-06 16:55:35 +03:00
* puint = ( version < < 16 ) + ( patchlevel < < 8 ) + sublevel ;
return 0 ;
}
2016-01-07 14:41:53 +03:00
const char * perf_tip ( const char * dirpath )
{
struct strlist * tips ;
struct str_node * node ;
char * tip = NULL ;
struct strlist_config conf = {
2016-01-09 13:16:29 +03:00
. dirname = dirpath ,
. file_only = true ,
2016-01-07 14:41:53 +03:00
} ;
tips = strlist__new ( " tips.txt " , & conf ) ;
2016-01-09 13:16:29 +03:00
if ( tips = = NULL )
2017-04-12 09:49:16 +03:00
return errno = = ENOENT ? NULL :
" Tip: check path of tips.txt or get more memory! ;-p " ;
2016-01-09 13:16:29 +03:00
if ( strlist__nr_entries ( tips ) = = 0 )
2016-01-07 14:41:53 +03:00
goto out ;
node = strlist__entry ( tips , random ( ) % strlist__nr_entries ( tips ) ) ;
if ( asprintf ( & tip , " Tip: %s " , node - > s ) < 0 )
tip = ( char * ) " Tip: get more memory! ;-) " ;
out :
strlist__delete ( tips ) ;
return tip ;
}
2016-01-20 14:56:32 +03:00
bool is_regular_file ( const char * file )
{
struct stat st ;
if ( stat ( file , & st ) )
return false ;
return S_ISREG ( st . st_mode ) ;
}
2016-01-25 12:56:13 +03:00
int fetch_current_timestamp ( char * buf , size_t sz )
{
struct timeval tv ;
struct tm tm ;
char dt [ 32 ] ;
if ( gettimeofday ( & tv , NULL ) | | ! localtime_r ( & tv . tv_sec , & tm ) )
return - 1 ;
if ( ! strftime ( dt , sizeof ( dt ) , " %Y%m%d%H%M%S " , & tm ) )
return - 1 ;
scnprintf ( buf , sz , " %s%02u " , dt , ( unsigned ) tv . tv_usec / 10000 ) ;
return 0 ;
}
2016-02-24 14:20:44 +03:00
void print_binary ( unsigned char * data , size_t len ,
size_t bytes_per_line , print_binary_t printer ,
void * extra )
{
size_t i , j , mask ;
if ( ! printer )
return ;
bytes_per_line = roundup_pow_of_two ( bytes_per_line ) ;
mask = bytes_per_line - 1 ;
printer ( BINARY_PRINT_DATA_BEGIN , 0 , extra ) ;
for ( i = 0 ; i < len ; i + + ) {
if ( ( i & mask ) = = 0 ) {
printer ( BINARY_PRINT_LINE_BEGIN , - 1 , extra ) ;
printer ( BINARY_PRINT_ADDR , i , extra ) ;
}
printer ( BINARY_PRINT_NUM_DATA , data [ i ] , extra ) ;
if ( ( ( i & mask ) = = mask ) | | i = = len - 1 ) {
for ( j = 0 ; j < mask - ( i & mask ) ; j + + )
printer ( BINARY_PRINT_NUM_PAD , - 1 , extra ) ;
printer ( BINARY_PRINT_SEP , i , extra ) ;
for ( j = i & ~ mask ; j < = i ; j + + )
printer ( BINARY_PRINT_CHAR_DATA , data [ j ] , extra ) ;
for ( j = 0 ; j < mask - ( i & mask ) ; j + + )
printer ( BINARY_PRINT_CHAR_PAD , i , extra ) ;
printer ( BINARY_PRINT_LINE_END , - 1 , extra ) ;
}
}
printer ( BINARY_PRINT_DATA_END , - 1 , extra ) ;
}
2016-07-16 19:11:19 +03:00
int is_printable_array ( char * p , unsigned int len )
{
unsigned int i ;
if ( ! p | | ! len | | p [ len - 1 ] ! = 0 )
return 0 ;
len - - ;
for ( i = 0 ; i < len ; i + + ) {
if ( ! isprint ( p [ i ] ) & & ! isspace ( p [ i ] ) )
return 0 ;
}
return 1 ;
}
2017-01-09 12:51:55 +03:00
int unit_number__scnprintf ( char * buf , size_t size , u64 n )
{
char unit [ 4 ] = " BKMG " ;
int i = 0 ;
while ( ( ( n / 1024 ) > 1 ) & & ( i < 3 ) ) {
n / = 1024 ;
i + + ;
}
return scnprintf ( buf , size , " % " PRIu64 " %c " , n , unit [ i ] ) ;
}