2015-03-25 12:49:25 -07:00
/* Copyright (c) 2013-2015 PLUMgrid, http://plumgrid.com
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of version 2 of the GNU General Public
* License as published by the Free Software Foundation .
*/
# include <linux/skbuff.h>
# include <linux/netdevice.h>
# include <linux/version.h>
# include <uapi/linux/bpf.h>
2020-01-20 14:06:49 +01:00
# include <bpf/bpf_helpers.h>
# include <bpf/bpf_tracing.h>
2015-03-25 12:49:25 -07:00
2020-05-16 13:06:08 +09:00
struct {
__uint ( type , BPF_MAP_TYPE_HASH ) ;
__type ( key , long ) ;
__type ( value , u64 ) ;
__uint ( max_entries , 4096 ) ;
} my_map SEC ( " .maps " ) ;
2015-03-25 12:49:25 -07:00
/* kprobe is NOT a stable ABI. If kernel internals change this bpf+kprobe
* example will no longer be meaningful
*/
2019-02-27 02:52:26 -05:00
SEC ( " kprobe/blk_mq_start_request " )
2015-03-25 12:49:25 -07:00
int bpf_prog1 ( struct pt_regs * ctx )
{
2015-07-06 16:20:07 +02:00
long rq = PT_REGS_PARM1 ( ctx ) ;
2015-03-25 12:49:25 -07:00
u64 val = bpf_ktime_get_ns ( ) ;
bpf_map_update_elem ( & my_map , & rq , & val , BPF_ANY ) ;
return 0 ;
}
static unsigned int log2l ( unsigned long long n )
{
# define S(k) if (n >= (1ull << k)) { i += k; n >>= k; }
int i = - ( n = = 0 ) ;
S ( 32 ) ; S ( 16 ) ; S ( 8 ) ; S ( 4 ) ; S ( 2 ) ; S ( 1 ) ;
return i ;
# undef S
}
# define SLOTS 100
2020-05-16 13:06:08 +09:00
struct {
__uint ( type , BPF_MAP_TYPE_PERCPU_ARRAY ) ;
__uint ( key_size , sizeof ( u32 ) ) ;
__uint ( value_size , sizeof ( u64 ) ) ;
__uint ( max_entries , SLOTS ) ;
} lat_map SEC ( " .maps " ) ;
2015-03-25 12:49:25 -07:00
2022-09-11 21:03:30 +08:00
SEC ( " kprobe/__blk_account_io_done " )
2015-03-25 12:49:25 -07:00
int bpf_prog2 ( struct pt_regs * ctx )
{
2015-07-06 16:20:07 +02:00
long rq = PT_REGS_PARM1 ( ctx ) ;
2015-03-25 12:49:25 -07:00
u64 * value , l , base ;
u32 index ;
value = bpf_map_lookup_elem ( & my_map , & rq ) ;
if ( ! value )
return 0 ;
u64 cur_time = bpf_ktime_get_ns ( ) ;
u64 delta = cur_time - * value ;
bpf_map_delete_elem ( & my_map , & rq ) ;
/* the lines below are computing index = log10(delta)*10
* using integer arithmetic
* index = 29 ~ 1 usec
* index = 59 ~ 1 msec
* index = 89 ~ 1 sec
* index = 99 ~ 10 sec or more
* log10 ( x ) * 10 = log2 ( x ) * 10 / log2 ( 10 ) = log2 ( x ) * 3
*/
l = log2l ( delta ) ;
base = 1ll < < l ;
index = ( l * 64 + ( delta - base ) * 64 / base ) * 3 / 64 ;
if ( index > = SLOTS )
index = SLOTS - 1 ;
value = bpf_map_lookup_elem ( & lat_map , & index ) ;
if ( value )
2016-02-01 22:39:58 -08:00
* value + = 1 ;
2015-03-25 12:49:25 -07:00
return 0 ;
}
char _license [ ] SEC ( " license " ) = " GPL " ;
u32 _version SEC ( " version " ) = LINUX_VERSION_CODE ;