@ -27,17 +27,16 @@ pub enum RelocationError {
#[ error(transparent) ]
#[ error(transparent) ]
IOError ( #[ from ] io ::Error ) ,
IOError ( #[ from ] io ::Error ) ,
#[ error( " section `{name}` not found" ) ]
#[ error( " program not found" ) ]
SectionNotFound { name : String } ,
ProgramNotFound ,
#[ error( " invalid relocation access string {access_str} " ) ]
#[ error( " invalid relocation access string {access_str} " ) ]
InvalidAccessString { access_str : String } ,
InvalidAccessString { access_str : String } ,
#[ error( " invalid instruction index #{index} referenced by relocation #{relocation_number} in section `{section_name}` " ) ]
#[ error( " invalid instruction index #{index} referenced by relocation #{relocation_number} , the program contains {num_instructions} instructions " ) ]
InvalidInstructionIndex {
InvalidInstructionIndex {
index : usize ,
index : usize ,
num_instructions : usize ,
num_instructions : usize ,
section_name : String ,
relocation_number : usize ,
relocation_number : usize ,
} ,
} ,
@ -170,7 +169,6 @@ impl Object {
} ) ? ;
} ) ? ;
let mut candidates_cache = HashMap ::< u32 , Vec < Candidate > > ::new ( ) ;
let mut candidates_cache = HashMap ::< u32 , Vec < Candidate > > ::new ( ) ;
for ( sec_name_off , relos ) in btf_ext . relocations ( ) {
for ( sec_name_off , relos ) in btf_ext . relocations ( ) {
let section_name = local_btf . string_at ( * sec_name_off ) ? ;
let section_name = local_btf . string_at ( * sec_name_off ) ? ;
@ -180,96 +178,120 @@ impl Object {
continue ;
continue ;
}
}
let section_name = parts [ 1 ] ;
let section_name = parts [ 1 ] ;
let program = self . programs . get_mut ( section_name ) . ok_or_else ( | | {
let program = self
RelocationError ::SectionNotFound {
. programs
name : section_name . to_string ( ) ,
. get_mut ( section_name )
. ok_or ( BpfError ::RelocationError {
program_name : section_name . to_owned ( ) ,
error : Box ::new ( RelocationError ::ProgramNotFound ) ,
} ) ? ;
match relocate_btf_program (
program ,
relos ,
local_btf ,
& target_btf ,
& mut candidates_cache ,
) {
Ok ( _ ) = > { }
Err ( ErrorWrapper ::BtfError ( e ) ) = > return Err ( e ) ? ,
Err ( ErrorWrapper ::RelocationError ( error ) ) = > {
return Err ( BpfError ::RelocationError {
program_name : section_name . to_owned ( ) ,
error : Box ::new ( error ) ,
} )
}
}
} ) ? ;
}
}
for rel in relos {
Ok ( ( ) )
let instructions = & mut program . instructions ;
}
let ins_index = rel . ins_offset as usize / std ::mem ::size_of ::< bpf_insn > ( ) ;
}
if ins_index > = instructions . len ( ) {
return Err ( RelocationError ::InvalidInstructionIndex {
index : ins_index ,
num_instructions : instructions . len ( ) ,
section_name : section_name . to_string ( ) ,
relocation_number : rel . number ,
} ) ? ;
}
let local_ty = local_btf . type_by_id ( rel . type_id ) ? ;
fn relocate_btf_program < ' target > (
let local_name = & * local_btf . type_name ( local_ty ) ? . unwrap ( ) ;
program : & mut Program ,
let access_str = & * local_btf . string_at ( rel . access_str_offset ) ? ;
relos : & [ Relocation ] ,
let local_spec = AccessSpec ::new ( local_btf , rel . type_id , access_str , * rel ) ? ;
local_btf : & Btf ,
target_btf : & ' target Btf ,
let mut matches = match rel . kind {
candidates_cache : & mut HashMap < u32 , Vec < Candidate < ' target > > > ,
RelocationKind ::TypeIdLocal = > Vec ::new ( ) , // we don't need to look at target types to relocate this value
) -> Result < ( ) , ErrorWrapper > {
_ = > {
for rel in relos {
let candidates = match candidates_cache . get ( & rel . type_id ) {
let instructions = & mut program . instructions ;
Some ( cands ) = > cands ,
let ins_index = rel . ins_offset as usize / std ::mem ::size_of ::< bpf_insn > ( ) ;
None = > {
if ins_index > = instructions . len ( ) {
candidates_cache . insert (
return Err ( RelocationError ::InvalidInstructionIndex {
rel . type_id ,
index : ins_index ,
find_candidates ( local_ty , local_name , & target_btf ) ? ,
num_instructions : instructions . len ( ) ,
) ;
relocation_number : rel . number ,
candidates_cache . get ( & rel . type_id ) . unwrap ( )
} ) ? ;
}
}
} ;
let mut matches = Vec ::new ( ) ;
let local_ty = local_btf . type_by_id ( rel . type_id ) ? ;
for candidate in candidates {
let local_name = & * local_btf . type_name ( local_ty ) ? . unwrap ( ) ;
if let Some ( candidate_spec ) = match_candidate ( & local_spec , candidate ) ? {
let access_str = & * local_btf . string_at ( rel . access_str_offset ) ? ;
let comp_rel = ComputedRelocation ::new (
let local_spec = AccessSpec ::new ( local_btf , rel . type_id , access_str , * rel ) ? ;
rel ,
& local_spec ,
Some ( & candidate_spec ) ,
) ? ;
matches . push ( ( candidate . name . clone ( ) , candidate_spec , comp_rel ) ) ;
}
}
matches
let mut matches = match rel . kind {
RelocationKind ::TypeIdLocal = > Vec ::new ( ) , // we don't need to look at target types to relocate this value
_ = > {
let candidates = match candidates_cache . get ( & rel . type_id ) {
Some ( cands ) = > cands ,
None = > {
candidates_cache . insert (
rel . type_id ,
find_candidates ( local_ty , local_name , target_btf ) ? ,
) ;
candidates_cache . get ( & rel . type_id ) . unwrap ( )
}
}
} ;
} ;
let comp_rel = if ! matches . is_empty ( ) {
let mut matches = Vec ::new ( ) ;
let mut matches = matches . drain ( .. ) ;
for candidate in candidates {
let ( _ , target_spec , target_comp_rel ) = matches . next ( ) . unwrap ( ) ;
if let Some ( candidate_spec ) = match_candidate ( & local_spec , candidate ) ? {
let comp_rel =
// if there's more than one candidate, make sure that they all resolve to the
ComputedRelocation ::new ( rel , & local_spec , Some ( & candidate_spec ) ) ? ;
// same value, else the relocation is ambiguous and can't be applied
matches . push ( ( candidate . name . clone ( ) , candidate_spec , comp_rel ) ) ;
let conflicts = matches
. filter_map ( | ( cand_name , cand_spec , cand_comp_rel ) | {
if cand_spec . bit_offset ! = target_spec . bit_offset
| | cand_comp_rel . target . value ! = target_comp_rel . target . value
{
Some ( cand_name . clone ( ) )
} else {
None
}
} )
. collect ::< Vec < _ > > ( ) ;
if ! conflicts . is_empty ( ) {
return Err ( RelocationError ::ConflictingCandidates {
type_name : local_name . to_string ( ) ,
candidates : conflicts ,
} ) ? ;
}
}
target_comp_rel
}
} else {
// there are no candidate matches and therefore no target_spec. This might mean
// that matching failed, or that the relocation can be applied looking at local
// types only
ComputedRelocation ::new ( rel , & local_spec , None ) ?
} ;
comp_rel. apply ( program , rel , section_name , local_btf , & target_btf ) ? ;
matches
}
}
}
} ;
Ok ( ( ) )
let comp_rel = if ! matches . is_empty ( ) {
let mut matches = matches . drain ( .. ) ;
let ( _ , target_spec , target_comp_rel ) = matches . next ( ) . unwrap ( ) ;
// if there's more than one candidate, make sure that they all resolve to the
// same value, else the relocation is ambiguous and can't be applied
let conflicts = matches
. filter_map ( | ( cand_name , cand_spec , cand_comp_rel ) | {
if cand_spec . bit_offset ! = target_spec . bit_offset
| | cand_comp_rel . target . value ! = target_comp_rel . target . value
{
Some ( cand_name . clone ( ) )
} else {
None
}
} )
. collect ::< Vec < _ > > ( ) ;
if ! conflicts . is_empty ( ) {
return Err ( RelocationError ::ConflictingCandidates {
type_name : local_name . to_string ( ) ,
candidates : conflicts ,
} ) ? ;
}
target_comp_rel
} else {
// there are no candidate matches and therefore no target_spec. This might mean
// that matching failed, or that the relocation can be applied looking at local
// types only
ComputedRelocation ::new ( rel , & local_spec , None ) ?
} ;
comp_rel . apply ( program , rel , local_btf , & target_btf ) ? ;
}
}
Ok ( ( ) )
}
}
fn flavorless_name ( name : & str ) -> & str {
fn flavorless_name ( name : & str ) -> & str {
@ -306,7 +328,7 @@ fn find_candidates<'target>(
fn match_candidate < ' target > (
fn match_candidate < ' target > (
local_spec : & AccessSpec ,
local_spec : & AccessSpec ,
candidate : & ' target Candidate ,
candidate : & ' target Candidate ,
) -> Result < Option < AccessSpec < ' target > > , Bpf Error> {
) -> Result < Option < AccessSpec < ' target > > , ErrorWrappe r> {
let mut target_spec = AccessSpec {
let mut target_spec = AccessSpec {
btf : candidate . btf ,
btf : candidate . btf ,
root_type_id : candidate . type_id ,
root_type_id : candidate . type_id ,
@ -434,7 +456,7 @@ fn match_member<'local, 'target>(
target_btf : & ' target Btf ,
target_btf : & ' target Btf ,
target_id : u32 ,
target_id : u32 ,
target_spec : & mut AccessSpec < ' target > ,
target_spec : & mut AccessSpec < ' target > ,
) -> Result < Option < u32 > , Bpf Error> {
) -> Result < Option < u32 > , ErrorWrappe r> {
let local_ty = local_btf . type_by_id ( local_accessor . type_id ) ? ;
let local_ty = local_btf . type_by_id ( local_accessor . type_id ) ? ;
let local_member = match local_ty {
let local_member = match local_ty {
BtfType ::Struct ( _ , members ) | BtfType ::Union ( _ , members ) = > {
BtfType ::Struct ( _ , members ) | BtfType ::Union ( _ , members ) = > {
@ -519,7 +541,7 @@ impl<'a> AccessSpec<'a> {
root_type_id : u32 ,
root_type_id : u32 ,
spec : & str ,
spec : & str ,
relocation : Relocation ,
relocation : Relocation ,
) -> Result < AccessSpec < ' a > , Bpf Error> {
) -> Result < AccessSpec < ' a > , ErrorWrappe r> {
let parts = spec
let parts = spec
. split ( ":" )
. split ( ":" )
. map ( | s | s . parse ::< usize > ( ) )
. map ( | s | s . parse ::< usize > ( ) )
@ -724,7 +746,7 @@ impl ComputedRelocation {
rel : & Relocation ,
rel : & Relocation ,
local_spec : & AccessSpec ,
local_spec : & AccessSpec ,
target_spec : Option < & AccessSpec > ,
target_spec : Option < & AccessSpec > ,
) -> Result < ComputedRelocation , Bpf Error> {
) -> Result < ComputedRelocation , ErrorWrappe r> {
use RelocationKind ::* ;
use RelocationKind ::* ;
let ret = match rel . kind {
let ret = match rel . kind {
FieldByteOffset | FieldByteSize | FieldExists | FieldSigned | FieldLShift64
FieldByteOffset | FieldByteSize | FieldExists | FieldSigned | FieldLShift64
@ -749,10 +771,9 @@ impl ComputedRelocation {
& self ,
& self ,
program : & mut Program ,
program : & mut Program ,
rel : & Relocation ,
rel : & Relocation ,
section_name : & str ,
local_btf : & Btf ,
local_btf : & Btf ,
target_btf : & Btf ,
target_btf : & Btf ,
) -> Result < ( ) , Bpf Error> {
) -> Result < ( ) , ErrorWrappe r> {
let instructions = & mut program . instructions ;
let instructions = & mut program . instructions ;
let num_instructions = instructions . len ( ) ;
let num_instructions = instructions . len ( ) ;
let ins_index = rel . ins_offset as usize / std ::mem ::size_of ::< bpf_insn > ( ) ;
let ins_index = rel . ins_offset as usize / std ::mem ::size_of ::< bpf_insn > ( ) ;
@ -762,7 +783,6 @@ impl ComputedRelocation {
. ok_or ( RelocationError ::InvalidInstructionIndex {
. ok_or ( RelocationError ::InvalidInstructionIndex {
index : rel . ins_offset as usize ,
index : rel . ins_offset as usize ,
num_instructions ,
num_instructions ,
section_name : section_name . to_string ( ) ,
relocation_number : rel . number ,
relocation_number : rel . number ,
} ) ? ;
} ) ? ;
@ -840,7 +860,6 @@ impl ComputedRelocation {
RelocationError ::InvalidInstructionIndex {
RelocationError ::InvalidInstructionIndex {
index : ins_index + 1 ,
index : ins_index + 1 ,
num_instructions ,
num_instructions ,
section_name : section_name . to_string ( ) ,
relocation_number : rel . number ,
relocation_number : rel . number ,
} ,
} ,
) ? ;
) ? ;
@ -862,7 +881,7 @@ impl ComputedRelocation {
fn compute_enum_relocation (
fn compute_enum_relocation (
rel : & Relocation ,
rel : & Relocation ,
spec : Option < & AccessSpec > ,
spec : Option < & AccessSpec > ,
) -> Result < ComputedRelocationValue , Bpf Error> {
) -> Result < ComputedRelocationValue , ErrorWrappe r> {
use RelocationKind ::* ;
use RelocationKind ::* ;
let value = match rel . kind {
let value = match rel . kind {
EnumVariantExists = > spec . is_some ( ) as u32 ,
EnumVariantExists = > spec . is_some ( ) as u32 ,
@ -888,7 +907,7 @@ impl ComputedRelocation {
fn compute_field_relocation (
fn compute_field_relocation (
rel : & Relocation ,
rel : & Relocation ,
spec : Option < & AccessSpec > ,
spec : Option < & AccessSpec > ,
) -> Result < ComputedRelocationValue , Bpf Error> {
) -> Result < ComputedRelocationValue , ErrorWrappe r> {
use RelocationKind ::* ;
use RelocationKind ::* ;
if let FieldExists = rel . kind {
if let FieldExists = rel . kind {
@ -1013,7 +1032,7 @@ impl ComputedRelocation {
rel : & Relocation ,
rel : & Relocation ,
local_spec : & AccessSpec ,
local_spec : & AccessSpec ,
target_spec : Option < & AccessSpec > ,
target_spec : Option < & AccessSpec > ,
) -> Result < ComputedRelocationValue , Bpf Error> {
) -> Result < ComputedRelocationValue , ErrorWrappe r> {
use RelocationKind ::* ;
use RelocationKind ::* ;
let value = match rel . kind {
let value = match rel . kind {
TypeIdLocal = > local_spec . root_type_id ,
TypeIdLocal = > local_spec . root_type_id ,
@ -1037,3 +1056,14 @@ impl ComputedRelocation {
} )
} )
}
}
}
}
// this exists only to simplify propagating errors from relocate_btf() and to associate
// RelocationError(s) with their respective program name
#[ derive(Error, Debug) ]
enum ErrorWrapper {
#[ error(transparent) ]
BtfError ( #[ from ] BtfError ) ,
#[ error(transparent) ]
RelocationError ( #[ from ] RelocationError ) ,
}