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 .
*/
2023-08-18 18:01:12 +09:00
# include "vmlinux.h"
2015-03-25 12:49:25 -07:00
# include <linux/version.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
2023-08-18 18:01:16 +09:00
struct start_key {
dev_t dev ;
u32 _pad ;
sector_t sector ;
} ;
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
2023-08-18 18:01:16 +09:00
/* from /sys/kernel/tracing/events/block/block_io_start/format */
SEC ( " tracepoint/block/block_io_start " )
int bpf_prog1 ( struct trace_event_raw_block_rq * ctx )
2015-03-25 12:49:25 -07:00
{
u64 val = bpf_ktime_get_ns ( ) ;
2023-08-18 18:01:16 +09:00
struct start_key key = {
. dev = ctx - > dev ,
. sector = ctx - > sector
} ;
2015-03-25 12:49:25 -07:00
2023-08-18 18:01:16 +09:00
bpf_map_update_elem ( & my_map , & key , & val , BPF_ANY ) ;
2015-03-25 12:49:25 -07:00
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
2023-08-18 18:01:16 +09:00
/* from /sys/kernel/tracing/events/block/block_io_done/format */
SEC ( " tracepoint/block/block_io_done " )
int bpf_prog2 ( struct trace_event_raw_block_rq * ctx )
2015-03-25 12:49:25 -07:00
{
2023-08-18 18:01:16 +09:00
struct start_key key = {
. dev = ctx - > dev ,
. sector = ctx - > sector
} ;
2015-03-25 12:49:25 -07:00
u64 * value , l , base ;
u32 index ;
2023-08-18 18:01:16 +09:00
value = bpf_map_lookup_elem ( & my_map , & key ) ;
2015-03-25 12:49:25 -07:00
if ( ! value )
return 0 ;
u64 cur_time = bpf_ktime_get_ns ( ) ;
u64 delta = cur_time - * value ;
2023-08-18 18:01:16 +09:00
bpf_map_delete_elem ( & my_map , & key ) ;
2015-03-25 12:49:25 -07:00
/* 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 ;