2009-08-17 16:18:06 +02:00
/*
* Copyright ( C ) 2009 , Steven Rostedt < srostedt @ redhat . com >
*
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; version 2 of the License ( not later ! )
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*/
2010-02-04 16:46:42 +08:00
# define _FILE_OFFSET_BITS 64
2009-08-17 16:18:06 +02:00
# include <dirent.h>
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include <getopt.h>
# include <stdarg.h>
# include <sys/types.h>
# include <sys/stat.h>
# include <sys/wait.h>
# include <sys/mman.h>
# include <pthread.h>
# include <fcntl.h>
# include <unistd.h>
# include <errno.h>
2009-08-28 03:09:58 +02:00
# include "../perf.h"
2009-08-17 16:18:06 +02:00
# include "util.h"
# include "trace-event.h"
static int input_fd ;
static int read_page ;
int file_bigendian ;
int host_bigendian ;
static int long_size ;
static unsigned long page_size ;
2010-04-01 23:59:21 -05:00
static ssize_t calc_data_size ;
2010-05-01 01:41:20 -05:00
static bool repipe ;
2010-04-01 23:59:21 -05:00
2012-04-06 00:47:56 +02:00
static void * malloc_or_die ( int size )
{
void * ret ;
ret = malloc ( size ) ;
if ( ! ret )
die ( " malloc " ) ;
return ret ;
}
2010-04-01 23:59:21 -05:00
static int do_read ( int fd , void * buf , int size )
{
int rsize = size ;
while ( size ) {
int ret = read ( fd , buf , size ) ;
if ( ret < = 0 )
return - 1 ;
2010-05-01 01:41:20 -05:00
if ( repipe ) {
int retw = write ( STDOUT_FILENO , buf , ret ) ;
if ( retw < = 0 | | retw ! = ret )
die ( " repiping input file " ) ;
}
2010-04-01 23:59:21 -05:00
size - = ret ;
buf + = ret ;
}
return rsize ;
}
2009-08-17 16:18:06 +02:00
static int read_or_die ( void * data , int size )
{
int r ;
2010-04-01 23:59:21 -05:00
r = do_read ( input_fd , data , size ) ;
if ( r < = 0 )
2009-08-17 16:18:06 +02:00
die ( " reading input file (size expected=%d received=%d) " ,
size , r ) ;
2010-04-01 23:59:21 -05:00
if ( calc_data_size )
calc_data_size + = r ;
2009-08-17 16:18:06 +02:00
return r ;
}
2010-05-04 23:02:10 -05:00
/* If it fails, the next read will report it */
static void skip ( int size )
{
char buf [ BUFSIZ ] ;
int r ;
while ( size ) {
r = size > BUFSIZ ? BUFSIZ : size ;
read_or_die ( buf , r ) ;
size - = r ;
} ;
}
2012-06-27 13:08:42 -03:00
static unsigned int read4 ( struct pevent * pevent )
2009-08-17 16:18:06 +02:00
{
unsigned int data ;
read_or_die ( & data , 4 ) ;
2012-06-27 13:08:42 -03:00
return __data2host4 ( pevent , data ) ;
2009-08-17 16:18:06 +02:00
}
2012-06-27 13:08:42 -03:00
static unsigned long long read8 ( struct pevent * pevent )
2009-08-17 16:18:06 +02:00
{
unsigned long long data ;
read_or_die ( & data , 8 ) ;
2012-06-27 13:08:42 -03:00
return __data2host8 ( pevent , data ) ;
2009-08-17 16:18:06 +02:00
}
static char * read_string ( void )
{
char buf [ BUFSIZ ] ;
char * str = NULL ;
int size = 0 ;
2010-02-04 16:46:42 +08:00
off_t r ;
2010-04-01 23:59:21 -05:00
char c ;
2009-08-17 16:18:06 +02:00
for ( ; ; ) {
2010-04-01 23:59:21 -05:00
r = read ( input_fd , & c , 1 ) ;
2009-08-17 16:18:06 +02:00
if ( r < 0 )
die ( " reading input file " ) ;
if ( ! r )
die ( " no data " ) ;
2010-05-01 01:41:20 -05:00
if ( repipe ) {
int retw = write ( STDOUT_FILENO , & c , 1 ) ;
if ( retw < = 0 | | retw ! = r )
die ( " repiping input file string " ) ;
}
2010-04-01 23:59:21 -05:00
buf [ size + + ] = c ;
2009-08-17 16:18:06 +02:00
2010-04-01 23:59:21 -05:00
if ( ! c )
break ;
2009-08-17 16:18:06 +02:00
}
2010-04-01 23:59:21 -05:00
if ( calc_data_size )
calc_data_size + = size ;
str = malloc_or_die ( size ) ;
memcpy ( str , buf , size ) ;
2009-08-17 16:18:06 +02:00
return str ;
}
2012-06-27 13:08:42 -03:00
static void read_proc_kallsyms ( struct pevent * pevent )
2009-08-17 16:18:06 +02:00
{
unsigned int size ;
char * buf ;
2012-06-27 13:08:42 -03:00
size = read4 ( pevent ) ;
2009-08-17 16:18:06 +02:00
if ( ! size )
return ;
2009-12-06 20:10:49 +09:00
buf = malloc_or_die ( size + 1 ) ;
2009-08-17 16:18:06 +02:00
read_or_die ( buf , size ) ;
2009-12-06 20:10:49 +09:00
buf [ size ] = ' \0 ' ;
2009-08-17 16:18:06 +02:00
2012-06-27 13:08:42 -03:00
parse_proc_kallsyms ( pevent , buf , size ) ;
2009-08-17 16:18:06 +02:00
free ( buf ) ;
}
2012-06-27 13:08:42 -03:00
static void read_ftrace_printk ( struct pevent * pevent )
2009-08-17 16:18:06 +02:00
{
unsigned int size ;
char * buf ;
2012-06-27 13:08:42 -03:00
size = read4 ( pevent ) ;
2009-08-17 16:18:06 +02:00
if ( ! size )
return ;
buf = malloc_or_die ( size ) ;
read_or_die ( buf , size ) ;
2012-06-27 13:08:42 -03:00
parse_ftrace_printk ( pevent , buf , size ) ;
2009-08-17 16:18:06 +02:00
free ( buf ) ;
}
2012-06-27 13:08:42 -03:00
static void read_header_files ( struct pevent * pevent )
2009-08-17 16:18:06 +02:00
{
unsigned long long size ;
char * header_event ;
char buf [ BUFSIZ ] ;
read_or_die ( buf , 12 ) ;
if ( memcmp ( buf , " header_page " , 12 ) ! = 0 )
die ( " did not read header page " ) ;
2012-06-27 13:08:42 -03:00
size = read8 ( pevent ) ;
2010-05-01 03:08:46 +02:00
skip ( size ) ;
2009-08-17 16:18:06 +02:00
/*
* The size field in the page is of type long ,
* use that instead , since it represents the kernel .
*/
long_size = header_page_size_size ;
read_or_die ( buf , 13 ) ;
if ( memcmp ( buf , " header_event " , 13 ) ! = 0 )
die ( " did not read header event " ) ;
2012-06-27 13:08:42 -03:00
size = read8 ( pevent ) ;
2009-08-17 16:18:06 +02:00
header_event = malloc_or_die ( size ) ;
read_or_die ( header_event , size ) ;
free ( header_event ) ;
}
2012-06-27 13:08:42 -03:00
static void read_ftrace_file ( struct pevent * pevent , unsigned long long size )
2009-08-17 16:18:06 +02:00
{
char * buf ;
buf = malloc_or_die ( size ) ;
read_or_die ( buf , size ) ;
2012-06-27 13:08:42 -03:00
parse_ftrace_file ( pevent , buf , size ) ;
2009-08-17 16:18:06 +02:00
free ( buf ) ;
}
2012-06-27 13:08:42 -03:00
static void read_event_file ( struct pevent * pevent , char * sys ,
unsigned long long size )
2009-08-17 16:18:06 +02:00
{
char * buf ;
buf = malloc_or_die ( size ) ;
read_or_die ( buf , size ) ;
2012-06-27 13:08:42 -03:00
parse_event_file ( pevent , buf , size , sys ) ;
2009-08-17 16:18:06 +02:00
free ( buf ) ;
}
2012-06-27 13:08:42 -03:00
static void read_ftrace_files ( struct pevent * pevent )
2009-08-17 16:18:06 +02:00
{
unsigned long long size ;
int count ;
int i ;
2012-06-27 13:08:42 -03:00
count = read4 ( pevent ) ;
2009-08-17 16:18:06 +02:00
for ( i = 0 ; i < count ; i + + ) {
2012-06-27 13:08:42 -03:00
size = read8 ( pevent ) ;
read_ftrace_file ( pevent , size ) ;
2009-08-17 16:18:06 +02:00
}
}
2012-06-27 13:08:42 -03:00
static void read_event_files ( struct pevent * pevent )
2009-08-17 16:18:06 +02:00
{
unsigned long long size ;
char * sys ;
int systems ;
int count ;
int i , x ;
2012-06-27 13:08:42 -03:00
systems = read4 ( pevent ) ;
2009-08-17 16:18:06 +02:00
for ( i = 0 ; i < systems ; i + + ) {
sys = read_string ( ) ;
2012-06-27 13:08:42 -03:00
count = read4 ( pevent ) ;
2009-08-17 16:18:06 +02:00
for ( x = 0 ; x < count ; x + + ) {
2012-06-27 13:08:42 -03:00
size = read8 ( pevent ) ;
read_event_file ( pevent , sys , size ) ;
2009-08-17 16:18:06 +02:00
}
}
}
struct cpu_data {
unsigned long long offset ;
unsigned long long size ;
unsigned long long timestamp ;
2012-04-06 00:48:06 +02:00
struct pevent_record * next ;
2009-08-17 16:18:06 +02:00
char * page ;
int cpu ;
int index ;
int page_size ;
} ;
static struct cpu_data * cpu_data ;
static void update_cpu_data_index ( int cpu )
{
cpu_data [ cpu ] . offset + = page_size ;
cpu_data [ cpu ] . size - = page_size ;
cpu_data [ cpu ] . index = 0 ;
}
static void get_next_page ( int cpu )
{
2010-02-04 16:46:42 +08:00
off_t save_seek ;
off_t ret ;
2009-08-17 16:18:06 +02:00
if ( ! cpu_data [ cpu ] . page )
return ;
if ( read_page ) {
if ( cpu_data [ cpu ] . size < = page_size ) {
free ( cpu_data [ cpu ] . page ) ;
cpu_data [ cpu ] . page = NULL ;
return ;
}
update_cpu_data_index ( cpu ) ;
/* other parts of the code may expect the pointer to not move */
2010-02-04 16:46:42 +08:00
save_seek = lseek ( input_fd , 0 , SEEK_CUR ) ;
2009-08-17 16:18:06 +02:00
2010-02-04 16:46:42 +08:00
ret = lseek ( input_fd , cpu_data [ cpu ] . offset , SEEK_SET ) ;
if ( ret = = ( off_t ) - 1 )
2009-08-17 16:18:06 +02:00
die ( " failed to lseek " ) ;
ret = read ( input_fd , cpu_data [ cpu ] . page , page_size ) ;
if ( ret < 0 )
die ( " failed to read page " ) ;
/* reset the file pointer back */
2010-02-04 16:46:42 +08:00
lseek ( input_fd , save_seek , SEEK_SET ) ;
2009-08-17 16:18:06 +02:00
return ;
}
munmap ( cpu_data [ cpu ] . page , page_size ) ;
cpu_data [ cpu ] . page = NULL ;
if ( cpu_data [ cpu ] . size < = page_size )
return ;
update_cpu_data_index ( cpu ) ;
cpu_data [ cpu ] . page = mmap ( NULL , page_size , PROT_READ , MAP_PRIVATE ,
input_fd , cpu_data [ cpu ] . offset ) ;
if ( cpu_data [ cpu ] . page = = MAP_FAILED )
die ( " failed to mmap cpu %d at offset 0x%llx " ,
cpu , cpu_data [ cpu ] . offset ) ;
}
static unsigned int type_len4host ( unsigned int type_len_ts )
{
if ( file_bigendian )
return ( type_len_ts > > 27 ) & ( ( 1 < < 5 ) - 1 ) ;
else
return type_len_ts & ( ( 1 < < 5 ) - 1 ) ;
}
static unsigned int ts4host ( unsigned int type_len_ts )
{
if ( file_bigendian )
return type_len_ts & ( ( 1 < < 27 ) - 1 ) ;
else
return type_len_ts > > 5 ;
}
static int calc_index ( void * ptr , int cpu )
{
return ( unsigned long ) ptr - ( unsigned long ) cpu_data [ cpu ] . page ;
}
2012-06-27 13:08:42 -03:00
struct pevent_record * trace_peek_data ( struct pevent * pevent , int cpu )
2009-08-17 16:18:06 +02:00
{
2012-04-06 00:48:06 +02:00
struct pevent_record * data ;
2009-08-17 16:18:06 +02:00
void * page = cpu_data [ cpu ] . page ;
int idx = cpu_data [ cpu ] . index ;
void * ptr = page + idx ;
unsigned long long extend ;
unsigned int type_len_ts ;
unsigned int type_len ;
unsigned int delta ;
unsigned int length = 0 ;
if ( cpu_data [ cpu ] . next )
return cpu_data [ cpu ] . next ;
if ( ! page )
return NULL ;
if ( ! idx ) {
/* FIXME: handle header page */
if ( header_page_ts_size ! = 8 )
die ( " expected a long long type for timestamp " ) ;
2012-06-27 13:08:42 -03:00
cpu_data [ cpu ] . timestamp = data2host8 ( pevent , ptr ) ;
2009-08-17 16:18:06 +02:00
ptr + = 8 ;
switch ( header_page_size_size ) {
case 4 :
2012-06-27 13:08:42 -03:00
cpu_data [ cpu ] . page_size = data2host4 ( pevent , ptr ) ;
2009-08-17 16:18:06 +02:00
ptr + = 4 ;
break ;
case 8 :
2012-06-27 13:08:42 -03:00
cpu_data [ cpu ] . page_size = data2host8 ( pevent , ptr ) ;
2009-08-17 16:18:06 +02:00
ptr + = 8 ;
break ;
default :
die ( " bad long size " ) ;
}
ptr = cpu_data [ cpu ] . page + header_page_data_offset ;
}
read_again :
idx = calc_index ( ptr , cpu ) ;
if ( idx > = cpu_data [ cpu ] . page_size ) {
get_next_page ( cpu ) ;
2012-06-27 13:08:42 -03:00
return trace_peek_data ( pevent , cpu ) ;
2009-08-17 16:18:06 +02:00
}
2012-06-27 13:08:42 -03:00
type_len_ts = data2host4 ( pevent , ptr ) ;
2009-08-17 16:18:06 +02:00
ptr + = 4 ;
type_len = type_len4host ( type_len_ts ) ;
delta = ts4host ( type_len_ts ) ;
switch ( type_len ) {
case RINGBUF_TYPE_PADDING :
if ( ! delta )
die ( " error, hit unexpected end of page " ) ;
2012-06-27 13:08:42 -03:00
length = data2host4 ( pevent , ptr ) ;
2009-08-17 16:18:06 +02:00
ptr + = 4 ;
length * = 4 ;
ptr + = length ;
goto read_again ;
case RINGBUF_TYPE_TIME_EXTEND :
2012-06-27 13:08:42 -03:00
extend = data2host4 ( pevent , ptr ) ;
2009-08-17 16:18:06 +02:00
ptr + = 4 ;
extend < < = TS_SHIFT ;
extend + = delta ;
cpu_data [ cpu ] . timestamp + = extend ;
goto read_again ;
case RINGBUF_TYPE_TIME_STAMP :
ptr + = 12 ;
break ;
case 0 :
2012-06-27 13:08:42 -03:00
length = data2host4 ( pevent , ptr ) ;
2009-08-17 16:18:06 +02:00
ptr + = 4 ;
die ( " here! length=%d " , length ) ;
break ;
default :
length = type_len * 4 ;
break ;
}
cpu_data [ cpu ] . timestamp + = delta ;
data = malloc_or_die ( sizeof ( * data ) ) ;
memset ( data , 0 , sizeof ( * data ) ) ;
data - > ts = cpu_data [ cpu ] . timestamp ;
data - > size = length ;
data - > data = ptr ;
ptr + = length ;
cpu_data [ cpu ] . index = calc_index ( ptr , cpu ) ;
cpu_data [ cpu ] . next = data ;
return data ;
}
2012-06-27 13:08:42 -03:00
struct pevent_record * trace_read_data ( struct pevent * pevent , int cpu )
2009-08-17 16:18:06 +02:00
{
2012-04-06 00:48:06 +02:00
struct pevent_record * data ;
2009-08-17 16:18:06 +02:00
2012-06-27 13:08:42 -03:00
data = trace_peek_data ( pevent , cpu ) ;
2009-08-17 16:18:06 +02:00
cpu_data [ cpu ] . next = NULL ;
return data ;
}
2012-06-27 13:08:42 -03:00
ssize_t trace_report ( int fd , struct pevent * * ppevent , bool __repipe )
2009-08-17 16:18:06 +02:00
{
char buf [ BUFSIZ ] ;
char test [ ] = { 23 , 8 , 68 } ;
char * version ;
2009-09-12 10:08:34 +02:00
int show_version = 0 ;
2009-08-17 16:18:06 +02:00
int show_funcs = 0 ;
int show_printk = 0 ;
2010-04-01 23:59:21 -05:00
ssize_t size ;
calc_data_size = 1 ;
2010-05-01 01:41:20 -05:00
repipe = __repipe ;
2009-08-17 16:18:06 +02:00
2009-10-06 23:36:47 +02:00
input_fd = fd ;
2009-08-17 16:18:06 +02:00
read_or_die ( buf , 3 ) ;
if ( memcmp ( buf , test , 3 ) ! = 0 )
2009-11-21 14:31:26 -02:00
die ( " no trace data in the file " ) ;
2009-08-17 16:18:06 +02:00
read_or_die ( buf , 7 ) ;
if ( memcmp ( buf , " tracing " , 7 ) ! = 0 )
2009-11-21 14:31:26 -02:00
die ( " not a trace file (missing 'tracing' tag) " ) ;
2009-08-17 16:18:06 +02:00
version = read_string ( ) ;
2009-09-12 10:08:34 +02:00
if ( show_version )
printf ( " version = %s \n " , version ) ;
2009-08-17 16:18:06 +02:00
free ( version ) ;
read_or_die ( buf , 1 ) ;
file_bigendian = buf [ 0 ] ;
host_bigendian = bigendian ( ) ;
2012-06-27 13:08:42 -03:00
* ppevent = read_trace_init ( file_bigendian , host_bigendian ) ;
if ( * ppevent = = NULL )
die ( " read_trace_init failed " ) ;
2012-04-06 00:47:56 +02:00
2009-08-17 16:18:06 +02:00
read_or_die ( buf , 1 ) ;
long_size = buf [ 0 ] ;
2012-06-27 13:08:42 -03:00
page_size = read4 ( * ppevent ) ;
2009-08-17 16:18:06 +02:00
2012-06-27 13:08:42 -03:00
read_header_files ( * ppevent ) ;
2009-08-17 16:18:06 +02:00
2012-06-27 13:08:42 -03:00
read_ftrace_files ( * ppevent ) ;
read_event_files ( * ppevent ) ;
read_proc_kallsyms ( * ppevent ) ;
read_ftrace_printk ( * ppevent ) ;
2009-08-17 16:18:06 +02:00
2010-04-01 23:59:21 -05:00
size = calc_data_size - 1 ;
calc_data_size = 0 ;
2010-05-01 01:41:20 -05:00
repipe = false ;
2010-04-01 23:59:21 -05:00
2009-08-17 16:18:06 +02:00
if ( show_funcs ) {
2012-06-27 13:08:42 -03:00
pevent_print_funcs ( * ppevent ) ;
2010-04-01 23:59:21 -05:00
return size ;
2009-08-17 16:18:06 +02:00
}
if ( show_printk ) {
2012-06-27 13:08:42 -03:00
pevent_print_printk ( * ppevent ) ;
2010-04-01 23:59:21 -05:00
return size ;
2009-08-17 16:18:06 +02:00
}
2010-04-01 23:59:21 -05:00
return size ;
2009-08-17 16:18:06 +02:00
}