2018-04-25 21:18:03 +09:00
/* SPDX-License-Identifier: GPL-2.0 */
/*
* Traceprobe fetch helper inlines
*/
static nokprobe_inline void
fetch_store_raw ( unsigned long val , struct fetch_insn * code , void * buf )
{
switch ( code - > size ) {
case 1 :
* ( u8 * ) buf = ( u8 ) val ;
break ;
case 2 :
* ( u16 * ) buf = ( u16 ) val ;
break ;
case 4 :
* ( u32 * ) buf = ( u32 ) val ;
break ;
case 8 :
//TBD: 32bit signed
* ( u64 * ) buf = ( u64 ) val ;
break ;
default :
* ( unsigned long * ) buf = val ;
}
}
static nokprobe_inline void
fetch_apply_bitfield ( struct fetch_insn * code , void * buf )
{
switch ( code - > basesize ) {
case 1 :
* ( u8 * ) buf < < = code - > lshift ;
* ( u8 * ) buf > > = code - > rshift ;
break ;
case 2 :
* ( u16 * ) buf < < = code - > lshift ;
* ( u16 * ) buf > > = code - > rshift ;
break ;
case 4 :
* ( u32 * ) buf < < = code - > lshift ;
* ( u32 * ) buf > > = code - > rshift ;
break ;
case 8 :
* ( u64 * ) buf < < = code - > lshift ;
* ( u64 * ) buf > > = code - > rshift ;
break ;
}
}
2018-04-25 21:19:01 +09:00
/*
2018-04-25 21:19:59 +09:00
* These functions must be defined for each callsite .
2018-04-25 21:19:01 +09:00
* Return consumed dynamic data size ( > = 0 ) , or error ( < 0 ) .
* If dest is NULL , don ' t store result and return required dynamic data size .
*/
2018-04-25 21:18:03 +09:00
static int
2021-08-19 00:13:28 -04:00
process_fetch_insn ( struct fetch_insn * code , void * rec ,
2018-04-25 21:19:01 +09:00
void * dest , void * base ) ;
2018-04-25 21:19:59 +09:00
static nokprobe_inline int fetch_store_strlen ( unsigned long addr ) ;
static nokprobe_inline int
fetch_store_string ( unsigned long addr , void * dest , void * base ) ;
2019-05-15 14:38:30 +09:00
static nokprobe_inline int fetch_store_strlen_user ( unsigned long addr ) ;
static nokprobe_inline int
fetch_store_string_user ( unsigned long addr , void * dest , void * base ) ;
2018-04-25 21:19:59 +09:00
static nokprobe_inline int
probe_mem_read ( void * dest , void * src , size_t size ) ;
2019-05-15 14:38:42 +09:00
static nokprobe_inline int
probe_mem_read_user ( void * dest , void * src , size_t size ) ;
2018-04-25 21:19:59 +09:00
2022-11-14 13:47:56 +09:00
static nokprobe_inline int
fetch_store_symstrlen ( unsigned long addr )
{
char namebuf [ KSYM_SYMBOL_LEN ] ;
int ret ;
ret = sprint_symbol ( namebuf , addr ) ;
if ( ret < 0 )
return 0 ;
return ret + 1 ;
}
/*
* Fetch a null - terminated symbol string + offset . Caller MUST set * ( u32 * ) buf
* with max length and relative data location .
*/
static nokprobe_inline int
fetch_store_symstring ( unsigned long addr , void * dest , void * base )
{
int maxlen = get_loc_len ( * ( u32 * ) dest ) ;
void * __dest ;
if ( unlikely ( ! maxlen ) )
return - ENOMEM ;
__dest = get_loc_data ( dest , base ) ;
return sprint_symbol ( __dest , addr ) ;
}
2022-12-30 14:33:53 +08:00
/* common part of process_fetch_insn*/
static nokprobe_inline int
process_common_fetch_insn ( struct fetch_insn * code , unsigned long * val )
{
switch ( code - > op ) {
case FETCH_OP_IMM :
* val = code - > immediate ;
break ;
case FETCH_OP_COMM :
* val = ( unsigned long ) current - > comm ;
break ;
case FETCH_OP_DATA :
* val = ( unsigned long ) code - > data ;
break ;
default :
return - EILSEQ ;
}
return 0 ;
}
2018-04-25 21:19:59 +09:00
/* From the 2nd stage, routine is same */
static nokprobe_inline int
process_fetch_insn_bottom ( struct fetch_insn * code , unsigned long val ,
void * dest , void * base )
{
2018-04-25 21:21:55 +09:00
struct fetch_insn * s3 = NULL ;
int total = 0 , ret = 0 , i = 0 ;
u32 loc = 0 ;
unsigned long lval = val ;
2018-04-25 21:19:59 +09:00
2018-04-25 21:21:55 +09:00
stage2 :
2018-04-25 21:19:59 +09:00
/* 2nd stage: dereference memory if needed */
2019-05-15 14:38:42 +09:00
do {
if ( code - > op = = FETCH_OP_DEREF ) {
lval = val ;
ret = probe_mem_read ( & val , ( void * ) val + code - > offset ,
sizeof ( val ) ) ;
} else if ( code - > op = = FETCH_OP_UDEREF ) {
lval = val ;
ret = probe_mem_read_user ( & val ,
( void * ) val + code - > offset , sizeof ( val ) ) ;
} else
break ;
2018-04-25 21:19:59 +09:00
if ( ret )
return ret ;
code + + ;
2019-05-15 14:38:42 +09:00
} while ( 1 ) ;
2018-04-25 21:19:59 +09:00
2018-04-25 21:21:55 +09:00
s3 = code ;
stage3 :
2018-04-25 21:19:59 +09:00
/* 3rd stage: store value to buffer */
if ( unlikely ( ! dest ) ) {
2022-11-14 13:47:56 +09:00
switch ( code - > op ) {
case FETCH_OP_ST_STRING :
2019-05-07 22:55:52 +09:00
ret = fetch_store_strlen ( val + code - > offset ) ;
2018-04-25 21:21:55 +09:00
code + + ;
goto array ;
2022-11-14 13:47:56 +09:00
case FETCH_OP_ST_USTRING :
2019-05-15 14:38:30 +09:00
ret + = fetch_store_strlen_user ( val + code - > offset ) ;
code + + ;
goto array ;
2022-11-14 13:47:56 +09:00
case FETCH_OP_ST_SYMSTR :
ret + = fetch_store_symstrlen ( val + code - > offset ) ;
code + + ;
goto array ;
default :
2018-04-25 21:19:59 +09:00
return - EILSEQ ;
2022-11-14 13:47:56 +09:00
}
2018-04-25 21:19:59 +09:00
}
switch ( code - > op ) {
case FETCH_OP_ST_RAW :
fetch_store_raw ( val , code , dest ) ;
break ;
case FETCH_OP_ST_MEM :
probe_mem_read ( dest , ( void * ) val + code - > offset , code - > size ) ;
break ;
2019-05-15 14:38:42 +09:00
case FETCH_OP_ST_UMEM :
probe_mem_read_user ( dest , ( void * ) val + code - > offset , code - > size ) ;
break ;
2018-04-25 21:19:59 +09:00
case FETCH_OP_ST_STRING :
2018-04-25 21:21:55 +09:00
loc = * ( u32 * ) dest ;
2018-04-25 21:19:59 +09:00
ret = fetch_store_string ( val + code - > offset , dest , base ) ;
break ;
2019-05-15 14:38:30 +09:00
case FETCH_OP_ST_USTRING :
loc = * ( u32 * ) dest ;
ret = fetch_store_string_user ( val + code - > offset , dest , base ) ;
break ;
2022-11-14 13:47:56 +09:00
case FETCH_OP_ST_SYMSTR :
loc = * ( u32 * ) dest ;
ret = fetch_store_symstring ( val + code - > offset , dest , base ) ;
break ;
2018-04-25 21:19:59 +09:00
default :
return - EILSEQ ;
}
code + + ;
/* 4th stage: modify stored value if needed */
if ( code - > op = = FETCH_OP_MOD_BF ) {
fetch_apply_bitfield ( code , dest ) ;
code + + ;
}
2018-04-25 21:21:55 +09:00
array :
/* the last stage: Loop on array */
if ( code - > op = = FETCH_OP_LP_ARRAY ) {
total + = ret ;
if ( + + i < code - > param ) {
code = s3 ;
2019-05-15 14:38:30 +09:00
if ( s3 - > op ! = FETCH_OP_ST_STRING & &
s3 - > op ! = FETCH_OP_ST_USTRING ) {
2018-04-25 21:21:55 +09:00
dest + = s3 - > size ;
val + = s3 - > size ;
goto stage3 ;
}
code - - ;
val = lval + sizeof ( char * ) ;
if ( dest ) {
dest + = sizeof ( u32 ) ;
* ( u32 * ) dest = update_data_loc ( loc , ret ) ;
}
goto stage2 ;
}
code + + ;
ret = total ;
}
2018-04-25 21:19:59 +09:00
return code - > op = = FETCH_OP_END ? ret : - EILSEQ ;
}
2018-04-25 21:18:03 +09:00
2021-03-23 18:49:35 +01:00
/* Sum up total data length for dynamic arrays (strings) */
2018-04-25 21:18:03 +09:00
static nokprobe_inline int
__get_data_size ( struct trace_probe * tp , struct pt_regs * regs )
{
struct probe_arg * arg ;
2018-04-25 21:19:01 +09:00
int i , len , ret = 0 ;
2018-04-25 21:18:03 +09:00
for ( i = 0 ; i < tp - > nr_args ; i + + ) {
arg = tp - > args + i ;
if ( unlikely ( arg - > dynamic ) ) {
2018-04-25 21:19:01 +09:00
len = process_fetch_insn ( arg - > code , regs , NULL , NULL ) ;
if ( len > 0 )
ret + = len ;
2018-04-25 21:18:03 +09:00
}
}
return ret ;
}
/* Store the value of each argument */
static nokprobe_inline void
2021-08-19 00:13:28 -04:00
store_trace_args ( void * data , struct trace_probe * tp , void * rec ,
2018-04-25 21:19:01 +09:00
int header_size , int maxlen )
2018-04-25 21:18:03 +09:00
{
struct probe_arg * arg ;
2018-04-25 21:19:01 +09:00
void * base = data - header_size ;
void * dyndata = data + tp - > size ;
u32 * dl ; /* Data location */
int ret , i ;
2018-04-25 21:18:03 +09:00
for ( i = 0 ; i < tp - > nr_args ; i + + ) {
arg = tp - > args + i ;
2018-04-25 21:19:01 +09:00
dl = data + arg - > offset ;
/* Point the dynamic data area if needed */
if ( unlikely ( arg - > dynamic ) )
* dl = make_data_loc ( maxlen , dyndata - base ) ;
2021-08-19 00:13:28 -04:00
ret = process_fetch_insn ( arg - > code , rec , dl , base ) ;
2019-02-06 20:00:13 +01:00
if ( unlikely ( ret < 0 & & arg - > dynamic ) ) {
2018-04-25 21:19:01 +09:00
* dl = make_data_loc ( 0 , dyndata - base ) ;
2019-02-06 20:00:13 +01:00
} else {
2018-04-25 21:19:01 +09:00
dyndata + = ret ;
2019-02-06 20:00:13 +01:00
maxlen - = ret ;
}
2018-04-25 21:18:03 +09:00
}
}