2012-04-06 00:47:56 +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
*
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*/
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include <errno.h>
2009-08-28 03:09:58 +02:00
# include "../perf.h"
2017-06-27 11:08:14 -03:00
# include "debug.h"
2009-08-17 16:18:07 +02:00
# include "trace-event.h"
2017-04-17 16:10:49 -03:00
# include "sane_ctype.h"
2012-04-06 00:47:56 +02:00
static int get_common_field ( struct scripting_context * context ,
int * offset , int * size , const char * type )
2010-01-27 02:27:54 -06:00
{
2018-08-08 14:02:46 -04:00
struct tep_handle * pevent = context - > pevent ;
2018-09-19 14:56:44 -04:00
struct tep_event_format * event ;
2018-09-19 14:56:45 -04:00
struct tep_format_field * field ;
2012-04-06 00:47:56 +02:00
if ( ! * size ) {
2018-10-05 12:22:25 -04:00
event = tep_get_first_event ( pevent ) ;
if ( ! event )
2012-04-06 00:47:56 +02:00
return 0 ;
2018-08-08 14:02:50 -04:00
field = tep_find_common_field ( event , type ) ;
2012-04-06 00:47:56 +02:00
if ( ! field )
return 0 ;
* offset = field - > offset ;
* size = field - > size ;
}
2018-08-08 14:02:53 -04:00
return tep_read_number ( pevent , context - > event_data + * offset , * size ) ;
2010-01-27 02:27:54 -06:00
}
int common_lock_depth ( struct scripting_context * context )
{
2012-04-06 00:47:56 +02:00
static int offset ;
static int size ;
int ret ;
ret = get_common_field ( context , & size , & offset ,
" common_lock_depth " ) ;
if ( ret < 0 )
return - 1 ;
return ret ;
}
int common_flags ( struct scripting_context * context )
{
static int offset ;
static int size ;
int ret ;
ret = get_common_field ( context , & size , & offset ,
" common_flags " ) ;
if ( ret < 0 )
return - 1 ;
return ret ;
}
int common_pc ( struct scripting_context * context )
{
static int offset ;
static int size ;
int ret ;
ret = get_common_field ( context , & size , & offset ,
" common_preempt_count " ) ;
if ( ret < 0 )
return - 1 ;
return ret ;
}
unsigned long long
2018-09-19 14:56:44 -04:00
raw_field_value ( struct tep_event_format * event , const char * name , void * data )
2012-04-06 00:47:56 +02:00
{
2018-09-19 14:56:45 -04:00
struct tep_format_field * field ;
2012-04-06 00:47:56 +02:00
unsigned long long val ;
2018-08-08 14:02:50 -04:00
field = tep_find_any_field ( event , name ) ;
2012-04-06 00:47:56 +02:00
if ( ! field )
return 0ULL ;
2018-08-08 14:02:53 -04:00
tep_read_number_field ( field , data , & val ) ;
2012-04-06 00:47:56 +02:00
return val ;
}
2018-09-19 14:56:44 -04:00
unsigned long long read_size ( struct tep_event_format * event , void * ptr , int size )
2012-04-06 00:47:56 +02:00
{
2018-08-08 14:02:53 -04:00
return tep_read_number ( event - > pevent , ptr , size ) ;
2012-04-06 00:47:56 +02:00
}
2018-09-19 14:56:44 -04:00
void event_format__fprintf ( struct tep_event_format * event ,
2015-02-03 12:46:58 -03:00
int cpu , void * data , int size , FILE * fp )
2012-04-06 00:47:56 +02:00
{
2018-08-08 14:02:47 -04:00
struct tep_record record ;
2012-04-06 00:47:56 +02:00
struct trace_seq s ;
memset ( & record , 0 , sizeof ( record ) ) ;
record . cpu = cpu ;
record . size = size ;
record . data = data ;
trace_seq_init ( & s ) ;
2018-08-08 14:02:49 -04:00
tep_event_info ( & s , event , & record ) ;
2015-02-03 12:46:58 -03:00
trace_seq_do_fprintf ( & s , fp ) ;
2014-02-02 22:38:49 +01:00
trace_seq_destroy ( & s ) ;
2012-04-06 00:47:56 +02:00
}
2018-09-19 14:56:44 -04:00
void event_format__print ( struct tep_event_format * event ,
2015-02-03 12:46:58 -03:00
int cpu , void * data , int size )
{
return event_format__fprintf ( event , cpu , data , size , stdout ) ;
}
2018-08-08 14:02:46 -04:00
void parse_ftrace_printk ( struct tep_handle * pevent ,
2012-09-11 01:15:03 +03:00
char * file , unsigned int size __maybe_unused )
2012-04-06 00:47:56 +02:00
{
unsigned long long addr ;
char * printk ;
char * line ;
char * next = NULL ;
char * addr_str ;
2015-05-26 11:41:37 -03:00
char * fmt = NULL ;
2012-04-06 00:47:56 +02:00
line = strtok_r ( file , " \n " , & next ) ;
while ( line ) {
addr_str = strtok_r ( line , " : " , & fmt ) ;
if ( ! addr_str ) {
2017-06-27 11:08:14 -03:00
pr_warning ( " printk format with empty entry " ) ;
2012-04-06 00:47:56 +02:00
break ;
}
addr = strtoull ( addr_str , NULL , 16 ) ;
/* fmt still has a space, skip it */
printk = strdup ( fmt + 1 ) ;
line = strtok_r ( NULL , " \n " , & next ) ;
2018-08-08 14:02:54 -04:00
tep_register_print_string ( pevent , printk , addr ) ;
2018-10-02 10:29:12 -04:00
free ( printk ) ;
2012-04-06 00:47:56 +02:00
}
}
2018-08-08 14:02:46 -04:00
void parse_saved_cmdline ( struct tep_handle * pevent ,
2013-04-11 17:25:04 +09:00
char * file , unsigned int size __maybe_unused )
{
2018-08-29 19:19:50 -07:00
char comm [ 17 ] ; /* Max comm length in the kernel is 16. */
2013-04-11 17:25:04 +09:00
char * line ;
char * next = NULL ;
int pid ;
line = strtok_r ( file , " \n " , & next ) ;
while ( line ) {
2018-08-29 19:19:50 -07:00
if ( sscanf ( line , " %d %16s " , & pid , comm ) = = 2 )
tep_register_comm ( pevent , comm , pid ) ;
2013-04-11 17:25:04 +09:00
line = strtok_r ( NULL , " \n " , & next ) ;
}
}
2018-08-08 14:02:46 -04:00
int parse_ftrace_file ( struct tep_handle * pevent , char * buf , unsigned long size )
2012-04-06 00:47:56 +02:00
{
2018-08-08 14:02:51 -04:00
return tep_parse_event ( pevent , buf , size , " ftrace " ) ;
2012-04-06 00:47:56 +02:00
}
2018-08-08 14:02:46 -04:00
int parse_event_file ( struct tep_handle * pevent ,
2012-06-27 13:08:42 -03:00
char * buf , unsigned long size , char * sys )
2012-04-06 00:47:56 +02:00
{
2018-08-08 14:02:51 -04:00
return tep_parse_event ( pevent , buf , size , sys ) ;
2012-04-06 00:47:56 +02:00
}
2018-09-19 14:56:44 -04:00
struct tep_event_format * trace_find_next_event ( struct tep_handle * pevent ,
struct tep_event_format * event )
2012-04-06 00:47:56 +02:00
{
static int idx ;
2018-10-05 12:22:25 -04:00
int events_count ;
struct tep_event_format * all_events ;
2012-04-06 00:47:56 +02:00
2018-10-05 12:22:25 -04:00
all_events = tep_get_first_event ( pevent ) ;
events_count = tep_get_events_count ( pevent ) ;
if ( ! pevent | | ! all_events | | events_count < 1 )
2012-04-06 00:47:56 +02:00
return NULL ;
if ( ! event ) {
idx = 0 ;
2018-10-05 12:22:25 -04:00
return all_events ;
2012-04-06 00:47:56 +02:00
}
2018-10-05 12:22:25 -04:00
if ( idx < events_count & & event = = ( all_events + idx ) ) {
2012-04-06 00:47:56 +02:00
idx + + ;
2018-10-05 12:22:25 -04:00
if ( idx = = events_count )
2012-04-06 00:47:56 +02:00
return NULL ;
2018-10-05 12:22:25 -04:00
return ( all_events + idx ) ;
2012-04-06 00:47:56 +02:00
}
2018-10-05 12:22:25 -04:00
for ( idx = 1 ; idx < events_count ; idx + + ) {
if ( event = = ( all_events + ( idx - 1 ) ) )
return ( all_events + idx ) ;
2012-04-06 00:47:56 +02:00
}
return NULL ;
}
struct flag {
const char * name ;
unsigned long long value ;
} ;
static const struct flag flags [ ] = {
{ " HI_SOFTIRQ " , 0 } ,
{ " TIMER_SOFTIRQ " , 1 } ,
{ " NET_TX_SOFTIRQ " , 2 } ,
{ " NET_RX_SOFTIRQ " , 3 } ,
{ " BLOCK_SOFTIRQ " , 4 } ,
2015-11-10 14:56:14 +01:00
{ " IRQ_POLL_SOFTIRQ " , 5 } ,
2012-04-06 00:47:56 +02:00
{ " TASKLET_SOFTIRQ " , 6 } ,
{ " SCHED_SOFTIRQ " , 7 } ,
{ " HRTIMER_SOFTIRQ " , 8 } ,
{ " RCU_SOFTIRQ " , 9 } ,
{ " HRTIMER_NORESTART " , 0 } ,
{ " HRTIMER_RESTART " , 1 } ,
} ;
unsigned long long eval_flag ( const char * flag )
{
int i ;
/*
* Some flags in the format files do not get converted .
* If the flag is not numeric , see if it is something that
* we already know about .
*/
if ( isdigit ( flag [ 0 ] ) )
return strtoull ( flag , NULL , 0 ) ;
for ( i = 0 ; i < ( int ) ( sizeof ( flags ) / sizeof ( flags [ 0 ] ) ) ; i + + )
if ( strcmp ( flags [ i ] . name , flag ) = = 0 )
return flags [ i ] . value ;
return 0 ;
2010-01-27 02:27:54 -06:00
}