@ -1,6 +1,6 @@
#![ no_std ]
#![ no_std ]
use core ::{ cmp, me m, ptr , slice } ;
use core ::{ mem, nu m, ptr , slice } ;
use num_enum ::IntoPrimitive ;
use num_enum ::IntoPrimitive ;
@ -8,8 +8,10 @@ pub const LOG_BUF_CAPACITY: usize = 8192;
pub const LOG_FIELDS : usize = 6 ;
pub const LOG_FIELDS : usize = 6 ;
#[ repr(usize) ]
pub type LogValueLength = u16 ;
#[ derive(Copy, Clone, Eq, PartialEq, Debug, Hash) ]
#[ repr(u8) ]
#[ derive(Copy, Clone, Eq, PartialEq, Debug, Hash, IntoPrimitive) ]
pub enum Level {
pub enum Level {
/// The "error" level.
/// The "error" level.
///
///
@ -33,7 +35,7 @@ pub enum Level {
Trace ,
Trace ,
}
}
#[ repr(u size )]
#[ repr(u 8 )]
#[ derive(Copy, Clone, Debug) ]
#[ derive(Copy, Clone, Debug) ]
pub enum RecordField {
pub enum RecordField {
Target = 1 ,
Target = 1 ,
@ -46,7 +48,7 @@ pub enum RecordField {
/// Types which are supported by aya-log and can be safely sent from eBPF
/// Types which are supported by aya-log and can be safely sent from eBPF
/// programs to userspace.
/// programs to userspace.
#[ repr(u size )]
#[ repr(u 8 )]
#[ derive(Copy, Clone, Debug) ]
#[ derive(Copy, Clone, Debug) ]
pub enum Argument {
pub enum Argument {
DisplayHint ,
DisplayHint ,
@ -111,26 +113,29 @@ where
}
}
pub ( crate ) fn write ( & self , mut buf : & mut [ u8 ] ) -> Result < usize , ( ) > {
pub ( crate ) fn write ( & self , mut buf : & mut [ u8 ] ) -> Result < usize , ( ) > {
let size = mem ::size_of ::< T > ( ) + mem ::size_of ::< usize > ( ) + self . value . len ( ) ;
// Break the abstraction to please the verifier.
let remaining = cmp ::min ( buf . len ( ) , LOG_BUF_CAPACITY ) ;
if buf . len ( ) > LOG_BUF_CAPACITY {
// Check if the size doesn't exceed the buffer bounds.
buf = & mut buf [ .. LOG_BUF_CAPACITY ] ;
if size > remaining {
}
let Self { tag , value } = self ;
let len = value . len ( ) ;
let wire_len : LogValueLength = value
. len ( )
. try_into ( )
. map_err ( | num ::TryFromIntError { .. } | ( ) ) ? ;
let size = mem ::size_of_val ( tag ) + mem ::size_of_val ( & wire_len ) + len ;
if size > buf . len ( ) {
return Err ( ( ) ) ;
return Err ( ( ) ) ;
}
}
unsafe { ptr ::write_unaligned ( buf . as_mut_ptr ( ) as * mut _ , self . tag ) } ;
unsafe { ptr ::write_unaligned ( buf . as_mut_ptr ( ) as * mut _ , * tag ) } ;
buf = & mut buf [ mem ::size_of ::< T > ( ) .. ] ;
buf = & mut buf [ mem ::size_of _val( tag ) .. ] ;
unsafe { ptr ::write_unaligned ( buf . as_mut_ptr ( ) as * mut _ , self . value . len ( ) ) } ;
unsafe { ptr ::write_unaligned ( buf . as_mut_ptr ( ) as * mut _ , wire_len ) } ;
buf = & mut buf [ mem ::size_of ::< usize > ( ) .. ] ;
buf = & mut buf [ mem ::size_of_val ( & wire_len ) .. ] ;
unsafe { ptr ::copy_nonoverlapping ( value . as_ptr ( ) , buf . as_mut_ptr ( ) , len ) } ;
let len = cmp ::min ( buf . len ( ) , self . value . len ( ) ) ;
// The verifier isn't happy with `len` being unbounded, so compare it
// with `LOG_BUF_CAPACITY`.
if len > LOG_BUF_CAPACITY {
return Err ( ( ) ) ;
}
buf [ .. len ] . copy_from_slice ( & self . value [ .. len ] ) ;
Ok ( size )
Ok ( size )
}
}
}
}
@ -210,10 +215,11 @@ pub fn write_record_header(
line : u32 ,
line : u32 ,
num_args : usize ,
num_args : usize ,
) -> Result < usize , ( ) > {
) -> Result < usize , ( ) > {
let level : u8 = level . into ( ) ;
let mut size = 0 ;
let mut size = 0 ;
for attr in [
for attr in [
TagLenValue ::< RecordField > ::new ( RecordField ::Target , target . as_bytes ( ) ) ,
TagLenValue ::< RecordField > ::new ( RecordField ::Target , target . as_bytes ( ) ) ,
TagLenValue ::< RecordField > ::new ( RecordField ::Level , & ( level as usize ) . to_ne_bytes ( ) ) ,
TagLenValue ::< RecordField > ::new ( RecordField ::Level , & level . to_ne_bytes ( ) ) ,
TagLenValue ::< RecordField > ::new ( RecordField ::Module , module . as_bytes ( ) ) ,
TagLenValue ::< RecordField > ::new ( RecordField ::Module , module . as_bytes ( ) ) ,
TagLenValue ::< RecordField > ::new ( RecordField ::File , file . as_bytes ( ) ) ,
TagLenValue ::< RecordField > ::new ( RecordField ::File , file . as_bytes ( ) ) ,
TagLenValue ::< RecordField > ::new ( RecordField ::Line , & line . to_ne_bytes ( ) ) ,
TagLenValue ::< RecordField > ::new ( RecordField ::Line , & line . to_ne_bytes ( ) ) ,
@ -224,3 +230,17 @@ pub fn write_record_header(
Ok ( size )
Ok ( size )
}
}
#[ cfg(test) ]
mod test {
use super ::* ;
fn log_value_length_sufficient ( ) {
assert! (
LOG_BUF_CAPACITY > = LogValueLength ::MAX . into ( ) ,
"{} < {}" ,
LOG_BUF_CAPACITY ,
LogValueLength ::MAX
) ;
}
}