2015-12-17 20:56:48 +03:00
/*
* Copyright ( c ) 2013 Ben Noordhuis < info @ bnoordhuis . nl >
* Copyright ( c ) 2013 - 2015 Dmitry V . Levin < ldv @ altlinux . org >
2016-09-03 14:44:48 +03:00
* Copyright ( c ) 2016 Eugene Syromyatnikov < evgsyr @ gmail . com >
2018-02-14 01:00:00 +03:00
* Copyright ( c ) 2015 - 2018 The strace developers .
2015-12-17 20:56:48 +03:00
* All rights reserved .
*
2018-12-10 03:00:00 +03:00
* SPDX - License - Identifier : LGPL - 2.1 - or - later
2015-12-17 20:56:48 +03:00
*/
2015-08-01 23:28:21 +03:00
# include "defs.h"
2016-09-03 14:44:48 +03:00
# include "perf_event_struct.h"
2018-08-31 07:05:08 +03:00
# include "print_fields.h"
2015-08-01 23:28:21 +03:00
2016-09-03 14:44:48 +03:00
# include "xlat/hw_breakpoint_len.h"
# include "xlat/hw_breakpoint_type.h"
# include "xlat/perf_attr_size.h"
# include "xlat/perf_branch_sample_type.h"
2015-08-01 23:28:21 +03:00
# include "xlat/perf_event_open_flags.h"
2016-09-03 14:44:48 +03:00
# include "xlat/perf_event_read_format.h"
# include "xlat/perf_event_sample_format.h"
# include "xlat/perf_hw_cache_id.h"
# include "xlat/perf_hw_cache_op_id.h"
# include "xlat/perf_hw_cache_op_result_id.h"
# include "xlat/perf_hw_id.h"
# include "xlat/perf_sw_ids.h"
# include "xlat/perf_type_id.h"
struct pea_desc {
struct perf_event_attr * attr ;
uint32_t size ;
} ;
static void
free_pea_desc ( void * pea_desc_ptr )
{
struct pea_desc * desc = pea_desc_ptr ;
free ( desc - > attr ) ;
free ( desc ) ;
}
2018-05-15 19:45:28 +03:00
int
2016-12-26 13:26:03 +03:00
fetch_perf_event_attr ( struct tcb * const tcp , const kernel_ulong_t addr )
2016-09-03 14:44:48 +03:00
{
struct pea_desc * desc ;
struct perf_event_attr * attr ;
uint32_t size ;
if ( umove ( tcp , addr + offsetof ( struct perf_event_attr , size ) , & size ) ) {
printaddr ( addr ) ;
return 1 ;
}
if ( size > sizeof ( * attr ) )
size = sizeof ( * attr ) ;
if ( ! size )
size = PERF_ATTR_SIZE_VER0 ;
/*
* Kernel ( rightfully ) deems invalid attribute structures with size less
* than first published format size , and we do the same .
*/
if ( size < PERF_ATTR_SIZE_VER0 ) {
printaddr ( addr ) ;
return 1 ;
}
if ( abbrev ( tcp ) )
size = offsetofend ( struct perf_event_attr , config ) ;
/* Size should be multiple of 8, but kernel doesn't check for it */
/* size &= ~7; */
attr = xcalloc ( 1 , sizeof ( * attr ) ) ;
if ( umoven_or_printaddr ( tcp , addr , size , attr ) ) {
free ( attr ) ;
return 1 ;
}
desc = xmalloc ( sizeof ( * desc ) ) ;
desc - > attr = attr ;
desc - > size = size ;
set_tcb_priv_data ( tcp , desc , free_pea_desc ) ;
return 0 ;
}
2018-05-15 19:45:28 +03:00
void
2016-12-26 13:26:03 +03:00
print_perf_event_attr ( struct tcb * const tcp , const kernel_ulong_t addr )
2016-09-03 14:44:48 +03:00
{
static const char * precise_ip_desc [ ] = {
" arbitrary skid " ,
" constant skid " ,
" requested to have 0 skid " ,
" must have 0 skid " ,
} ;
struct pea_desc * desc ;
struct perf_event_attr * attr ;
uint32_t size ;
uint32_t new_size ;
int use_new_size = 0 ;
/*
* Amusingly , kernel accepts structures with only part of the field
* present , so we making check like this ( instead of checking
* offsetofend against size ) in order to print fields as kernel sees
* them . This also should work great on big endian architectures .
*/
# define _PERF_CHECK_FIELD(_field) \
do { \
if ( offsetof ( struct perf_event_attr , _field ) > = size ) \
goto print_perf_event_attr_out ; \
} while ( 0 )
desc = get_tcb_priv_data ( tcp ) ;
attr = desc - > attr ;
size = desc - > size ;
/* The only error which expected to change size field currently */
if ( tcp - > u_error = = E2BIG ) {
if ( umove ( tcp , addr + offsetof ( struct perf_event_attr , size ) ,
& new_size ) )
use_new_size = - 1 ;
else
use_new_size = 1 ;
}
2018-08-31 07:05:08 +03:00
PRINT_FIELD_XVAL ( " { " , * attr , type , perf_type_id , " PERF_TYPE_??? " ) ;
PRINT_FIELD_XVAL ( " , " , * attr , size , perf_attr_size ,
" PERF_ATTR_SIZE_??? " ) ;
2016-09-03 14:44:48 +03:00
if ( use_new_size ) {
tprints ( " => " ) ;
if ( use_new_size > 0 )
printxval ( perf_attr_size , new_size ,
2017-06-18 01:23:09 +03:00
" PERF_ATTR_SIZE_??? " ) ;
2016-09-03 14:44:48 +03:00
else
tprints ( " ??? " ) ;
}
switch ( attr - > type ) {
case PERF_TYPE_HARDWARE :
2018-08-31 07:05:08 +03:00
PRINT_FIELD_XVAL ( " , " , * attr , config , perf_hw_id ,
" PERF_COUNT_HW_??? " ) ;
2016-09-03 14:44:48 +03:00
break ;
case PERF_TYPE_SOFTWARE :
2018-08-31 07:05:08 +03:00
PRINT_FIELD_XVAL ( " , " , * attr , config , perf_sw_ids ,
" PERF_COUNT_SW_??? " ) ;
2016-09-03 14:44:48 +03:00
break ;
case PERF_TYPE_TRACEPOINT :
/*
* " The value to use in config can be obtained from under
2017-06-18 01:23:09 +03:00
* debugfs tracing / events / . . / . . / id if ftrace is enabled
* in the kernel . "
2016-09-03 14:44:48 +03:00
*/
2018-08-31 07:05:08 +03:00
PRINT_FIELD_U ( " , " , * attr , config ) ;
2016-09-03 14:44:48 +03:00
break ;
case PERF_TYPE_HW_CACHE :
/*
* ( perf_hw_cache_id ) | ( perf_hw_cache_op_id < < 8 ) |
* ( perf_hw_cache_op_result_id < < 16 )
*/
2018-08-31 07:05:08 +03:00
tprints ( " , config= " ) ;
printxval ( perf_hw_cache_id , attr - > config & 0xFF ,
" PERF_COUNT_HW_CACHE_??? " ) ;
tprints ( " | " ) ;
printxval ( perf_hw_cache_op_id , ( attr - > config > > 8 ) & 0xFF ,
2017-06-18 01:23:09 +03:00
" PERF_COUNT_HW_CACHE_OP_??? " ) ;
2018-08-31 07:05:08 +03:00
tprints ( " <<8| " ) ;
2016-09-03 14:44:48 +03:00
/*
* Current code ( see set_ext_hw_attr in arch / x86 / events / core . c ,
* tile_map_cache_event in arch / tile / kernel / perf_event . c ,
* arc_pmu_cache_event in arch / arc / kernel / perf_event . c ,
* hw_perf_cache_event in arch / blackfin / kernel / perf_event . c ,
* _hw_perf_cache_event in arch / metag / kernel / perf / perf_event . c ,
* mipspmu_map_cache_event in arch / mips / kernel / perf_event_mipsxx . c ,
* hw_perf_cache_event in arch / powerpc / perf / core - book3s . c ,
* hw_perf_cache_event in arch / powerpc / perf / core - fsl - emb . c ,
* hw_perf_cache_event in arch / sh / kernel / perf_event . c ,
* sparc_map_cache_event in arch / sparc / kernel / perf_event . c ,
* xtensa_pmu_cache_event in arch / xtensa / kernel / perf_event . c ,
* armpmu_map_cache_event in drivers / perf / arm_pmu . c ) assumes
* that cache result is 8 bits in size .
*/
2018-08-31 07:05:08 +03:00
printxval ( perf_hw_cache_op_result_id ,
( attr - > config > > 16 ) & 0xFF ,
" PERF_COUNT_HW_CACHE_RESULT_??? " ) ;
2017-02-18 03:58:17 +03:00
tprints ( " <<16 " ) ;
Introduce tprintf_comment and tprints_comment functions
* defs.h (tprintf_comment, tprints_comment): New prototypes.
* strace.c (tvprintf): New function.
(tprintf): Use it.
(tprintf_comment, tprints_comment): New functions.
* aio.c (tprint_lio_opcode): Use tprints_comment.
* dm.c (dm_decode_dm_target_spec, dm_decode_dm_target_deps,
dm_decode_dm_name_list, dm_decode_dm_target_versions,
dm_decode_dm_target_msg, dm_decode_string, dm_known_ioctl): Likewise.
* futex.c (SYS_FUNC(futex)): Likewise.
* perf.c (print_perf_event_attr): Likewise.
* seccomp.c (decode_bpf_code): Likewise.
* util.c (printxvals, printxval_searchn, printflags64): Likewise.
* btrfs.c (print_u64, btrfs_print_key_type, btrfs_print_objectid,
print_key_value_internal): Likewise.
(btrfs_ioctl): Use tprints_comment and tprintf_comment.
* dirent.c (SYS_FUNC(getdents)): Likewise.
* dirent64.c (SYS_FUNC(getdents64)): Likewise.
* execve.c (printargc): Use tprintf_comment.
* tests/btrfs.c (btrfs_test_get_dev_stats_ioctl,
btrfs_test_features_ioctls): Update expected output.
2017-04-24 22:31:54 +03:00
if ( attr - > config > > 24 ) {
tprintf ( " |%# " PRIx64 " <<24 " , attr - > config > > 24 ) ;
tprints_comment ( " PERF_COUNT_HW_CACHE_??? " ) ;
}
2016-09-03 14:44:48 +03:00
break ;
case PERF_TYPE_RAW :
/*
* " If type is PERF_TYPE_RAW, then a custom " raw " config
* value is needed . Most CPUs support events that are not
* covered by the " generalized " events . These are
* implementation defined ; see your CPU manual ( for example the
* Intel Volume 3 B documentation or the AMD BIOS and Kernel
* Developer Guide ) . The libpfm4 library can be used to
* translate from the name in the architectural manuals
* to the raw hex value perf_event_open ( ) expects in this
* field . "
*/
case PERF_TYPE_BREAKPOINT :
/*
* " If type is PERF_TYPE_BREAKPOINT, then leave config set
* to zero . Its parameters are set in other places . "
*/
default :
2018-08-31 07:05:08 +03:00
PRINT_FIELD_X ( " , " , * attr , config ) ;
2016-09-03 14:44:48 +03:00
break ;
}
if ( abbrev ( tcp ) )
goto print_perf_event_attr_out ;
if ( attr - > freq )
2018-08-31 07:05:08 +03:00
PRINT_FIELD_U ( " , " , * attr , sample_freq ) ;
2016-09-03 14:44:48 +03:00
else
2018-08-31 07:05:08 +03:00
PRINT_FIELD_U ( " , " , * attr , sample_period ) ;
2016-09-03 14:44:48 +03:00
2018-08-31 07:05:08 +03:00
PRINT_FIELD_FLAGS ( " , " , * attr , sample_type , perf_event_sample_format ,
" PERF_SAMPLE_??? " ) ;
PRINT_FIELD_FLAGS ( " , " , * attr , read_format , perf_event_read_format ,
" PERF_FORMAT_??? " ) ;
2016-09-03 14:44:48 +03:00
tprintf ( " , disabled=%u "
2017-06-18 01:23:09 +03:00
" , inherit=%u "
" , pinned=%u "
" , exclusive=%u "
" , exclusive_user=%u "
" , exclude_kernel=%u "
" , exclude_hv=%u "
" , exclude_idle=%u "
" , mmap=%u "
" , comm=%u "
" , freq=%u "
" , inherit_stat=%u "
" , enable_on_exec=%u "
" , task=%u "
" , watermark=%u "
" , precise_ip=%u " ,
attr - > disabled ,
attr - > inherit ,
attr - > pinned ,
attr - > exclusive ,
attr - > exclude_user ,
attr - > exclude_kernel ,
attr - > exclude_hv ,
attr - > exclude_idle ,
attr - > mmap ,
attr - > comm ,
attr - > freq ,
attr - > inherit_stat ,
attr - > enable_on_exec ,
attr - > task ,
attr - > watermark ,
attr - > precise_ip ) ;
Introduce tprintf_comment and tprints_comment functions
* defs.h (tprintf_comment, tprints_comment): New prototypes.
* strace.c (tvprintf): New function.
(tprintf): Use it.
(tprintf_comment, tprints_comment): New functions.
* aio.c (tprint_lio_opcode): Use tprints_comment.
* dm.c (dm_decode_dm_target_spec, dm_decode_dm_target_deps,
dm_decode_dm_name_list, dm_decode_dm_target_versions,
dm_decode_dm_target_msg, dm_decode_string, dm_known_ioctl): Likewise.
* futex.c (SYS_FUNC(futex)): Likewise.
* perf.c (print_perf_event_attr): Likewise.
* seccomp.c (decode_bpf_code): Likewise.
* util.c (printxvals, printxval_searchn, printflags64): Likewise.
* btrfs.c (print_u64, btrfs_print_key_type, btrfs_print_objectid,
print_key_value_internal): Likewise.
(btrfs_ioctl): Use tprints_comment and tprintf_comment.
* dirent.c (SYS_FUNC(getdents)): Likewise.
* dirent64.c (SYS_FUNC(getdents64)): Likewise.
* execve.c (printargc): Use tprintf_comment.
* tests/btrfs.c (btrfs_test_get_dev_stats_ioctl,
btrfs_test_features_ioctls): Update expected output.
2017-04-24 22:31:54 +03:00
tprints_comment ( precise_ip_desc [ attr - > precise_ip ] ) ;
tprintf ( " , mmap_data=%u "
2017-06-18 01:23:09 +03:00
" , sample_id_all=%u "
" , exclude_host=%u "
" , exclude_guest=%u "
" , exclude_callchain_kernel=%u "
" , exclude_callchain_user=%u "
" , mmap2=%u "
" , comm_exec=%u "
" , use_clockid=%u "
" , context_switch=%u "
2018-01-08 21:48:42 +03:00
" , write_backward=%u "
" , namespaces=%u " ,
2017-06-18 01:23:09 +03:00
attr - > mmap_data ,
attr - > sample_id_all ,
attr - > exclude_host ,
attr - > exclude_guest ,
attr - > exclude_callchain_kernel ,
attr - > exclude_callchain_user ,
attr - > mmap2 ,
attr - > comm_exec ,
attr - > use_clockid ,
attr - > context_switch ,
2018-01-08 21:48:42 +03:00
attr - > write_backward ,
attr - > namespaces ) ;
2016-09-03 14:44:48 +03:00
/*
* Print it only in case it is non - zero , since it may contain flags we
* are not aware about .
*/
Introduce tprintf_comment and tprints_comment functions
* defs.h (tprintf_comment, tprints_comment): New prototypes.
* strace.c (tvprintf): New function.
(tprintf): Use it.
(tprintf_comment, tprints_comment): New functions.
* aio.c (tprint_lio_opcode): Use tprints_comment.
* dm.c (dm_decode_dm_target_spec, dm_decode_dm_target_deps,
dm_decode_dm_name_list, dm_decode_dm_target_versions,
dm_decode_dm_target_msg, dm_decode_string, dm_known_ioctl): Likewise.
* futex.c (SYS_FUNC(futex)): Likewise.
* perf.c (print_perf_event_attr): Likewise.
* seccomp.c (decode_bpf_code): Likewise.
* util.c (printxvals, printxval_searchn, printflags64): Likewise.
* btrfs.c (print_u64, btrfs_print_key_type, btrfs_print_objectid,
print_key_value_internal): Likewise.
(btrfs_ioctl): Use tprints_comment and tprintf_comment.
* dirent.c (SYS_FUNC(getdents)): Likewise.
* dirent64.c (SYS_FUNC(getdents64)): Likewise.
* execve.c (printargc): Use tprintf_comment.
* tests/btrfs.c (btrfs_test_get_dev_stats_ioctl,
btrfs_test_features_ioctls): Update expected output.
2017-04-24 22:31:54 +03:00
if ( attr - > __reserved_1 ) {
tprintf ( " , __reserved_1=%# " PRIx64 ,
2017-06-18 01:23:09 +03:00
( uint64_t ) attr - > __reserved_1 ) ;
2018-01-08 21:48:42 +03:00
tprints_comment ( " Bits 63..29 " ) ;
Introduce tprintf_comment and tprints_comment functions
* defs.h (tprintf_comment, tprints_comment): New prototypes.
* strace.c (tvprintf): New function.
(tprintf): Use it.
(tprintf_comment, tprints_comment): New functions.
* aio.c (tprint_lio_opcode): Use tprints_comment.
* dm.c (dm_decode_dm_target_spec, dm_decode_dm_target_deps,
dm_decode_dm_name_list, dm_decode_dm_target_versions,
dm_decode_dm_target_msg, dm_decode_string, dm_known_ioctl): Likewise.
* futex.c (SYS_FUNC(futex)): Likewise.
* perf.c (print_perf_event_attr): Likewise.
* seccomp.c (decode_bpf_code): Likewise.
* util.c (printxvals, printxval_searchn, printflags64): Likewise.
* btrfs.c (print_u64, btrfs_print_key_type, btrfs_print_objectid,
print_key_value_internal): Likewise.
(btrfs_ioctl): Use tprints_comment and tprintf_comment.
* dirent.c (SYS_FUNC(getdents)): Likewise.
* dirent64.c (SYS_FUNC(getdents64)): Likewise.
* execve.c (printargc): Use tprintf_comment.
* tests/btrfs.c (btrfs_test_get_dev_stats_ioctl,
btrfs_test_features_ioctls): Update expected output.
2017-04-24 22:31:54 +03:00
}
2016-09-03 14:44:48 +03:00
if ( attr - > watermark )
2018-08-31 07:05:08 +03:00
PRINT_FIELD_U ( " , " , * attr , wakeup_watermark ) ;
2016-09-03 14:44:48 +03:00
else
2018-08-31 07:05:08 +03:00
PRINT_FIELD_U ( " , " , * attr , wakeup_events ) ;
2016-09-03 14:44:48 +03:00
if ( attr - > type = = PERF_TYPE_BREAKPOINT )
/* Any combination of R/W with X is deemed invalid */
2018-08-31 07:05:08 +03:00
PRINT_FIELD_XVAL ( " , " , * attr , bp_type , hw_breakpoint_type ,
( attr - > bp_type < =
( HW_BREAKPOINT_X | HW_BREAKPOINT_RW ) )
? " HW_BREAKPOINT_INVALID "
: " HW_BREAKPOINT_??? " ) ;
2016-09-03 14:44:48 +03:00
if ( attr - > type = = PERF_TYPE_BREAKPOINT )
2018-08-31 07:05:08 +03:00
PRINT_FIELD_X ( " , " , * attr , bp_addr ) ;
2016-09-03 14:44:48 +03:00
else
2018-08-31 07:05:08 +03:00
PRINT_FIELD_X ( " , " , * attr , config1 ) ;
2016-09-03 14:44:48 +03:00
/*
* Fields after bp_addr / config1 are optional and may not present ; check
* against size is needed .
*/
_PERF_CHECK_FIELD ( bp_len ) ;
if ( attr - > type = = PERF_TYPE_BREAKPOINT )
2018-08-31 07:05:08 +03:00
PRINT_FIELD_U ( " , " , * attr , bp_len ) ;
2016-09-03 14:44:48 +03:00
else
2018-08-31 07:05:08 +03:00
PRINT_FIELD_X ( " , " , * attr , config2 ) ;
2016-09-03 14:44:48 +03:00
_PERF_CHECK_FIELD ( branch_sample_type ) ;
if ( attr - > sample_type & PERF_SAMPLE_BRANCH_STACK ) {
2018-08-31 07:05:08 +03:00
PRINT_FIELD_FLAGS ( " , " , * attr , branch_sample_type ,
perf_branch_sample_type ,
" PERF_SAMPLE_BRANCH_??? " ) ;
2016-09-03 14:44:48 +03:00
}
_PERF_CHECK_FIELD ( sample_regs_user ) ;
/*
* " This bit mask defines the set of user CPU registers to dump on
* samples . The layout of the register mask is architecture - specific and
* described in the kernel header
* arch / ARCH / include / uapi / asm / perf_regs . h . "
*/
2018-08-31 07:05:08 +03:00
PRINT_FIELD_X ( " , " , * attr , sample_regs_user ) ;
2016-09-03 14:44:48 +03:00
_PERF_CHECK_FIELD ( sample_stack_user ) ;
/*
* " size of the user stack to dump if PERF_SAMPLE_STACK_USER is
* specified . "
*/
if ( attr - > sample_type & PERF_SAMPLE_STACK_USER )
2018-08-31 07:05:08 +03:00
PRINT_FIELD_X ( " , " , * attr , sample_stack_user ) ;
2016-09-03 14:44:48 +03:00
if ( attr - > use_clockid ) {
_PERF_CHECK_FIELD ( clockid ) ;
2018-08-31 07:05:08 +03:00
PRINT_FIELD_XVAL ( " , " , * attr , clockid , clocknames , " CLOCK_??? " ) ;
2016-09-03 14:44:48 +03:00
}
_PERF_CHECK_FIELD ( sample_regs_intr ) ;
2018-08-31 07:05:08 +03:00
PRINT_FIELD_X ( " , " , * attr , sample_regs_intr ) ;
2016-09-03 14:44:48 +03:00
_PERF_CHECK_FIELD ( aux_watermark ) ;
2018-08-31 07:05:08 +03:00
PRINT_FIELD_U ( " , " , * attr , aux_watermark ) ;
2016-09-03 14:44:48 +03:00
_PERF_CHECK_FIELD ( sample_max_stack ) ;
2018-08-31 07:05:08 +03:00
PRINT_FIELD_U ( " , " , * attr , sample_max_stack ) ;
2016-09-03 14:44:48 +03:00
/* _PERF_CHECK_FIELD(__reserved_2);
2018-08-31 07:05:08 +03:00
PRINT_FIELD_U ( " , " , * attr , __reserved2 ) ; */
2016-09-03 14:44:48 +03:00
print_perf_event_attr_out :
if ( ( attr - > size & & ( attr - > size > size ) ) | |
( ! attr - > size & & ( size < PERF_ATTR_SIZE_VER0 ) ) )
2017-02-18 03:58:17 +03:00
tprints ( " , ... " ) ;
2016-09-03 14:44:48 +03:00
2017-02-18 03:58:17 +03:00
tprints ( " } " ) ;
2016-09-03 14:44:48 +03:00
}
2015-08-01 23:28:21 +03:00
SYS_FUNC ( perf_event_open )
{
2016-09-03 14:44:48 +03:00
/*
* We try to copy out the whole structure on entering in order to check
* size value on exiting . We do not check the rest of the fields because
* they shouldn ' t be changed , but copy the whole structure instead
* of just size field because they could .
*/
if ( entering ( tcp ) ) {
if ( ! fetch_perf_event_attr ( tcp , tcp - > u_arg [ 0 ] ) )
return 0 ;
} else {
print_perf_event_attr ( tcp , tcp - > u_arg [ 0 ] ) ;
}
2018-02-07 16:45:02 +03:00
tprintf ( " , %d, %d, " ,
2015-08-01 23:28:21 +03:00
( int ) tcp - > u_arg [ 1 ] ,
2018-02-07 16:45:02 +03:00
( int ) tcp - > u_arg [ 2 ] ) ;
printfd ( tcp , tcp - > u_arg [ 3 ] ) ;
tprints ( " , " ) ;
2016-12-26 00:55:01 +03:00
printflags64 ( perf_event_open_flags , tcp - > u_arg [ 4 ] , " PERF_FLAG_??? " ) ;
2015-08-01 23:28:21 +03:00
2015-08-03 12:06:59 +03:00
return RVAL_DECODED | RVAL_FD ;
2015-08-01 23:28:21 +03:00
}