2015-07-26 02:55:51 +03:00
/*
* Copyright ( c ) 2015 Dmitry V . Levin < ldv @ altlinux . org >
2017-01-26 17:36:56 +03:00
* Copyright ( c ) 2017 Quentin Monnet < quentin . monnet @ 6 wind . com >
2015-07-26 02:55:51 +03:00
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
* 2. Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
* 3. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission .
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ` ` AS IS ' ' AND ANY EXPRESS OR
* IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED .
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT , INDIRECT ,
* INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT
* NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*/
# include "defs.h"
# ifdef HAVE_LINUX_BPF_H
# include <linux / bpf.h>
# endif
# include "xlat/bpf_commands.h"
# include "xlat/bpf_map_types.h"
# include "xlat/bpf_prog_types.h"
# include "xlat/bpf_map_update_elem_flags.h"
2017-01-26 17:36:56 +03:00
# include "xlat/bpf_attach_type.h"
2017-02-17 03:05:00 +03:00
# include "xlat/bpf_attach_flags.h"
2015-07-26 02:55:51 +03:00
static int
2016-12-26 13:26:03 +03:00
bpf_map_create ( struct tcb * const tcp , const kernel_ulong_t addr ,
2016-12-21 05:43:47 +03:00
unsigned int size )
2015-07-26 02:55:51 +03:00
{
struct {
uint32_t map_type , key_size , value_size , max_entries ;
} attr = { } ;
if ( ! size ) {
printaddr ( addr ) ;
return RVAL_DECODED | RVAL_FD ;
}
if ( size > sizeof ( attr ) )
size = sizeof ( attr ) ;
if ( umoven_or_printaddr ( tcp , addr , size , & attr ) )
return RVAL_DECODED | RVAL_FD ;
tprints ( " {map_type= " ) ;
printxval ( bpf_map_types , attr . map_type , " BPF_MAP_TYPE_??? " ) ;
tprintf ( " , key_size=%u, value_size=%u, max_entries=%u} " ,
attr . key_size , attr . value_size , attr . max_entries ) ;
return RVAL_DECODED | RVAL_FD ;
}
static void
2016-12-26 13:26:03 +03:00
bpf_map_update_elem ( struct tcb * const tcp , const kernel_ulong_t addr ,
2016-12-21 05:43:47 +03:00
unsigned int size )
2015-07-26 02:55:51 +03:00
{
struct {
uint32_t map_fd ;
uint64_t ATTRIBUTE_ALIGNED ( 8 ) key ;
uint64_t ATTRIBUTE_ALIGNED ( 8 ) value ;
uint64_t flags ;
} attr = { } ;
if ( ! size ) {
printaddr ( addr ) ;
return ;
}
if ( size > sizeof ( attr ) )
size = sizeof ( attr ) ;
if ( umoven_or_printaddr ( tcp , addr , size , & attr ) )
return ;
tprints ( " {map_fd= " ) ;
printfd ( tcp , attr . map_fd ) ;
tprintf ( " , key=%# " PRIx64 " , value=%# " PRIx64 " , flags= " ,
attr . key , attr . value ) ;
2016-05-17 00:27:30 +03:00
printxval64 ( bpf_map_update_elem_flags , attr . flags , " BPF_??? " ) ;
2015-07-26 02:55:51 +03:00
tprints ( " } " ) ;
}
static void
2016-12-26 13:26:03 +03:00
bpf_map_delete_elem ( struct tcb * const tcp , const kernel_ulong_t addr ,
2016-12-21 05:43:47 +03:00
unsigned int size )
2015-07-26 02:55:51 +03:00
{
struct {
uint32_t map_fd ;
uint64_t ATTRIBUTE_ALIGNED ( 8 ) key ;
} attr = { } ;
if ( ! size ) {
printaddr ( addr ) ;
return ;
}
if ( size > sizeof ( attr ) )
size = sizeof ( attr ) ;
if ( umoven_or_printaddr ( tcp , addr , size , & attr ) )
return ;
tprints ( " {map_fd= " ) ;
printfd ( tcp , attr . map_fd ) ;
tprintf ( " , key=%# " PRIx64 " } " , attr . key ) ;
}
static int
2016-12-26 13:26:03 +03:00
bpf_map_io ( struct tcb * const tcp , const kernel_ulong_t addr , unsigned int size ,
2016-12-21 05:43:47 +03:00
const char * const text )
2015-07-26 02:55:51 +03:00
{
struct bpf_io_elem_struct {
uint32_t map_fd ;
uint64_t ATTRIBUTE_ALIGNED ( 8 ) key ;
uint64_t ATTRIBUTE_ALIGNED ( 8 ) value ;
} attr = { } ;
if ( exiting ( tcp ) ) {
if ( ! syserror ( tcp ) & & ! umove_or_printaddr ( tcp , addr , & attr ) )
tprintf ( " , %s=%# " PRIx64 , text , attr . value ) ;
tprints ( " } " ) ;
return RVAL_DECODED ;
}
if ( ! size ) {
printaddr ( addr ) ;
return RVAL_DECODED ;
}
if ( size > sizeof ( attr ) )
size = sizeof ( attr ) ;
if ( umoven_or_printaddr ( tcp , addr , size , & attr ) )
return RVAL_DECODED ;
tprints ( " {map_fd= " ) ;
printfd ( tcp , attr . map_fd ) ;
tprintf ( " , key=%# " PRIx64 , attr . key ) ;
return 0 ;
}
static int
2016-12-26 13:26:03 +03:00
bpf_prog_load ( struct tcb * const tcp , const kernel_ulong_t addr ,
2016-12-21 05:43:47 +03:00
unsigned int size )
2015-07-26 02:55:51 +03:00
{
struct {
uint32_t prog_type , insn_cnt ;
uint64_t ATTRIBUTE_ALIGNED ( 8 ) insns , license ;
uint32_t log_level , log_size ;
uint64_t ATTRIBUTE_ALIGNED ( 8 ) log_buf ;
uint32_t kern_version ;
} attr = { } ;
if ( ! size ) {
printaddr ( addr ) ;
return RVAL_DECODED | RVAL_FD ;
}
if ( size > sizeof ( attr ) )
size = sizeof ( attr ) ;
if ( umoven_or_printaddr ( tcp , addr , size , & attr ) )
return RVAL_DECODED | RVAL_FD ;
tprints ( " {prog_type= " ) ;
printxval ( bpf_prog_types , attr . prog_type , " BPF_PROG_TYPE_??? " ) ;
tprintf ( " , insn_cnt=%u, insns=%# " PRIx64 " , license= " ,
attr . insn_cnt , attr . insns ) ;
2016-12-20 19:43:26 +03:00
printstr ( tcp , attr . license ) ;
2015-07-26 02:55:51 +03:00
tprintf ( " , log_level=%u, log_size=%u, log_buf=%# " PRIx64 " , kern_version=%u} " ,
attr . log_level , attr . log_size , attr . log_buf , attr . kern_version ) ;
return RVAL_DECODED | RVAL_FD ;
}
2017-01-26 17:36:56 +03:00
static int
bpf_obj_manage ( struct tcb * const tcp , const kernel_ulong_t addr ,
unsigned int size )
{
struct {
uint64_t ATTRIBUTE_ALIGNED ( 8 ) pathname ;
uint32_t bpf_fd ;
} attr = { } ;
if ( ! size ) {
printaddr ( addr ) ;
return RVAL_DECODED | RVAL_FD ;
}
if ( size > sizeof ( attr ) )
size = sizeof ( attr ) ;
if ( umoven_or_printaddr ( tcp , addr , size , & attr ) )
return RVAL_DECODED | RVAL_FD ;
2017-02-18 03:58:17 +03:00
tprints ( " {pathname= " ) ;
2017-01-26 17:36:56 +03:00
printpath ( tcp , attr . pathname ) ;
tprints ( " , bpf_fd= " ) ;
printfd ( tcp , attr . bpf_fd ) ;
2017-02-18 03:58:17 +03:00
tprints ( " } " ) ;
2017-01-26 17:36:56 +03:00
return RVAL_DECODED | RVAL_FD ;
}
static int
2017-01-29 01:11:20 +03:00
bpf_prog_attach_detach ( struct tcb * const tcp , const kernel_ulong_t addr ,
2017-02-17 03:05:00 +03:00
unsigned int size , bool print_attach )
2017-01-26 17:36:56 +03:00
{
struct {
2017-02-17 03:05:00 +03:00
uint32_t target_fd , attach_bpf_fd , attach_type , attach_flags ;
2017-01-26 17:36:56 +03:00
} attr = { } ;
if ( ! size ) {
printaddr ( addr ) ;
return RVAL_DECODED ;
}
if ( size > sizeof ( attr ) )
size = sizeof ( attr ) ;
if ( umoven_or_printaddr ( tcp , addr , size , & attr ) )
return RVAL_DECODED ;
2017-02-18 03:58:17 +03:00
tprints ( " {target_fd= " ) ;
2017-01-26 17:36:56 +03:00
printfd ( tcp , attr . target_fd ) ;
2017-02-17 03:05:00 +03:00
if ( print_attach ) {
2017-02-18 03:58:17 +03:00
tprints ( " , attach_bpf_fd= " ) ;
2017-01-29 01:11:20 +03:00
printfd ( tcp , attr . attach_bpf_fd ) ;
}
2017-02-18 03:58:17 +03:00
tprints ( " , attach_type= " ) ;
2017-01-26 17:36:56 +03:00
printxval ( bpf_attach_type , attr . attach_type , " BPF_??? " ) ;
2017-02-17 03:05:00 +03:00
if ( print_attach ) {
2017-02-18 03:58:17 +03:00
tprints ( " , attach_flags= " ) ;
2017-02-17 03:05:00 +03:00
printflags ( bpf_attach_flags , attr . attach_flags , " BPF_F_??? " ) ;
}
2017-02-18 03:58:17 +03:00
tprints ( " } " ) ;
2017-01-26 17:36:56 +03:00
return RVAL_DECODED ;
}
static int
2017-01-29 01:11:20 +03:00
bpf_prog_attach ( struct tcb * const tcp , const kernel_ulong_t addr ,
2017-01-26 17:36:56 +03:00
unsigned int size )
{
2017-01-29 01:11:20 +03:00
return bpf_prog_attach_detach ( tcp , addr , size , true ) ;
}
2017-01-26 17:36:56 +03:00
2017-01-29 01:11:20 +03:00
static int
bpf_prog_detach ( struct tcb * const tcp , const kernel_ulong_t addr ,
unsigned int size )
{
return bpf_prog_attach_detach ( tcp , addr , size , false ) ;
2017-01-26 17:36:56 +03:00
}
2015-07-26 02:55:51 +03:00
SYS_FUNC ( bpf )
{
2016-12-21 05:43:47 +03:00
const unsigned int cmd = tcp - > u_arg [ 0 ] ;
2016-12-26 13:26:03 +03:00
const kernel_ulong_t addr = tcp - > u_arg [ 1 ] ;
2015-07-26 02:55:51 +03:00
const unsigned int size = tcp - > u_arg [ 2 ] ;
int rc = RVAL_DECODED ;
if ( entering ( tcp ) ) {
printxval ( bpf_commands , cmd , " BPF_??? " ) ;
tprints ( " , " ) ;
}
switch ( cmd ) {
case BPF_MAP_CREATE :
rc = bpf_map_create ( tcp , addr , size ) ;
break ;
case BPF_MAP_LOOKUP_ELEM :
rc = bpf_map_io ( tcp , addr , size , " value " ) ;
break ;
case BPF_MAP_UPDATE_ELEM :
bpf_map_update_elem ( tcp , addr , size ) ;
break ;
case BPF_MAP_DELETE_ELEM :
bpf_map_delete_elem ( tcp , addr , size ) ;
break ;
case BPF_MAP_GET_NEXT_KEY :
rc = bpf_map_io ( tcp , addr , size , " next_key " ) ;
break ;
case BPF_PROG_LOAD :
rc = bpf_prog_load ( tcp , addr , size ) ;
break ;
2017-01-26 17:36:56 +03:00
case BPF_OBJ_PIN :
rc = bpf_obj_manage ( tcp , addr , size ) ;
break ;
case BPF_OBJ_GET :
rc = bpf_obj_manage ( tcp , addr , size ) ;
break ;
case BPF_PROG_ATTACH :
rc = bpf_prog_attach ( tcp , addr , size ) ;
break ;
case BPF_PROG_DETACH :
rc = bpf_prog_detach ( tcp , addr , size ) ;
break ;
2015-07-26 02:55:51 +03:00
default :
printaddr ( addr ) ;
break ;
}
if ( rc & RVAL_DECODED )
tprintf ( " , %u " , size ) ;
return rc ;
}