LCOV - code coverage report
Current view: top level - src/elf - binary.rs (source / functions) Coverage Total Hit
Test: lief.lcov Lines: 87.1 % 233 203
Test Date: 2025-03-30:00:00:00 Functions: 66.7 % 78 52

            Line data    Source code
       1              : use std::mem::size_of;
       2              : use std::path::Path;
       3              : use std::pin::Pin;
       4              : 
       5              : use num_traits::{cast, Num};
       6              : 
       7              : use lief_ffi as ffi;
       8              : 
       9              : use super::builder::Config;
      10              : use super::dynamic::{self, DynamicEntries, DynamicEntry, Library};
      11              : use super::hash::{Gnu, Sysv};
      12              : use super::header::Header;
      13              : use super::note::ItNotes;
      14              : use super::relocation::{
      15              :     DynamicRelocations, ObjectRelocations, PltGotRelocations, Relocation, Relocations,
      16              : };
      17              : use super::section::{Section, Sections};
      18              : use super::segment::Segments;
      19              : use super::symbol::{DynamicSymbols, ExportedSymbols, ImportedSymbols, SymtabSymbols};
      20              : use super::symbol_versioning::{SymbolVersion, SymbolVersionDefinition, SymbolVersionRequirement};
      21              : use super::{Segment, Symbol};
      22              : use crate::Error;
      23              : 
      24              : use crate::common::{into_optional, FromFFI};
      25              : use crate::generic;
      26              : use crate::{declare_iterator, to_conv_result, to_result, to_slice};
      27              : 
      28            0 : #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
      29              : pub enum ElfClass {
      30              :     Elf32,
      31              :     Elf64,
      32              :     Unknown,
      33              : }
      34              : 
      35              : impl ElfClass {
      36              :     const ELF_CLASS32: u32 = 1;
      37              :     const ELF_CLASS64: u32 = 2;
      38              : 
      39            0 :     pub fn from_value(value: u32) -> Self {
      40            0 :         match value {
      41            0 :             Self::ELF_CLASS32 => ElfClass::Elf32,
      42            0 :             Self::ELF_CLASS64 => ElfClass::Elf64,
      43            0 :             _ => ElfClass::Unknown,
      44              :         }
      45            0 :     }
      46              : }
      47              : 
      48              : /// This is the main interface to read and write ELF binary attributes.
      49              : ///
      50              : /// Note that this structure implements the [`generic::Binary`] trait from which other generic
      51              : /// functions are exposed
      52              : pub struct Binary {
      53              :     ptr: cxx::UniquePtr<ffi::ELF_Binary>,
      54              : }
      55              : 
      56              : impl std::fmt::Debug for Binary {
      57           80 :     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
      58           80 :         f.debug_struct("Binary")
      59           80 :             .field("header", &self.header())
      60           80 :             .finish()
      61           80 :     }
      62              : }
      63              : 
      64              : impl FromFFI<ffi::ELF_Binary> for Binary {
      65          180 :     fn from_ffi(ptr: cxx::UniquePtr<ffi::ELF_Binary>) -> Self {
      66          180 :         Self { ptr }
      67          180 :     }
      68              : }
      69              : 
      70              : impl Binary {
      71              :     /// Create a [`Binary`] from the given file path
      72          100 :     pub fn parse(path: &str) -> Option<Self> {
      73          100 :         let bin = ffi::ELF_Binary::parse(path);
      74          100 :         if bin.is_null() {
      75            0 :             return None;
      76          100 :         }
      77          100 :         Some(Binary::from_ffi(bin))
      78          100 :     }
      79              : 
      80              :     /// Return the main ELF header
      81           80 :     pub fn header(&self) -> Header {
      82           80 :         Header::from_ffi(self.ptr.header())
      83           80 :     }
      84              : 
      85              :     /// Return the size taken by the binary when loaded (virtual size)
      86           80 :     pub fn virtual_size(&self) -> u64 {
      87           80 :         self.ptr.virtual_size()
      88           80 :     }
      89              : 
      90              :     /// Return the path to the ELF interpreter that is used to process the ELF information
      91              :     /// once loaded by the kernel
      92           80 :     pub fn interpreter(&self) -> String {
      93           80 :         self.ptr.interpreter().to_string()
      94           80 :     }
      95              : 
      96              :     /// Return sysv-hash information (if present)
      97           80 :     pub fn sysv_hash(&self) -> Option<Sysv> {
      98           80 :         into_optional(self.ptr.sysv_hash())
      99           80 :     }
     100              : 
     101              :     /// Return GNU Hash info (if present)
     102           80 :     pub fn gnu_hash(&self) -> Option<Gnu> {
     103           80 :         into_optional(self.ptr.gnu_hash())
     104           80 :     }
     105              : 
     106              :     /// Return an iterator over the [`crate::elf::Section`] of the binary
     107           80 :     pub fn sections(&self) -> Sections {
     108           80 :         Sections::new(self.ptr.sections())
     109           80 :     }
     110              : 
     111              :     /// Return an iterator over the [`crate::elf::Segment`] of the binary
     112           80 :     pub fn segments(&self) -> Segments {
     113           80 :         Segments::new(self.ptr.segments())
     114           80 :     }
     115              : 
     116              :     /// Return an iterator over the [`crate::elf::DynamicEntries`] of the binary
     117           80 :     pub fn dynamic_entries(&self) -> DynamicEntries {
     118           80 :         DynamicEntries::new(self.ptr.dynamic_entries())
     119           80 :     }
     120              : 
     121              :     /// Remove **all** dynamic entries with the given tag
     122            0 :     pub fn remove_dynamic_entries_by_tag(&mut self, tag: dynamic::Tag) {
     123            0 :         self.ptr.as_mut().unwrap().remove_dynamic_entries_by_tag(tag.into())
     124            0 :     }
     125              : 
     126              :     /// Add the given dynamic entry and return the new entry
     127            0 :     pub fn add_dynamic_entry(&mut self, entry: &dynamic::Entries) -> dynamic::Entries {
     128            0 :         dynamic::Entries::from_ffi(self.ptr.as_mut().unwrap().add_dynamic_entry(entry.as_base()))
     129            0 :     }
     130              : 
     131              :     /// Return an iterator over the dynamic [`crate::elf::Symbol`] of the binary
     132           80 :     pub fn dynamic_symbols(&self) -> DynamicSymbols {
     133           80 :         DynamicSymbols::new(self.ptr.dynamic_symbols())
     134           80 :     }
     135              : 
     136              :     /// Return an iterator over the **exported** [`crate::elf::Symbol`] of the binary
     137           80 :     pub fn exported_symbols(&self) -> ExportedSymbols {
     138           80 :         ExportedSymbols::new(self.ptr.exported_symbols())
     139           80 :     }
     140              : 
     141              :     /// Return an iterator over the **imported** [`crate::elf::Symbol`] of the binary
     142           80 :     pub fn imported_symbols(&self) -> ImportedSymbols {
     143           80 :         ImportedSymbols::new(self.ptr.imported_symbols())
     144           80 :     }
     145              : 
     146              :     /// Return an iterator over the symtab-debug [`crate::elf::Symbol`] of the binary
     147           80 :     pub fn symtab_symbols(&self) -> SymtabSymbols {
     148           80 :         SymtabSymbols::new(self.ptr.symtab_symbols())
     149           80 :     }
     150              : 
     151              :     /// Return an iterator over the  [`crate::elf::SymbolVersion`] of the binary
     152          160 :     pub fn symbols_version(&self) -> SymbolsVersion {
     153          160 :         SymbolsVersion::new(self.ptr.symbols_version())
     154          160 :     }
     155              : 
     156              :     /// Return an iterator over the  [`crate::elf::SymbolVersionRequirement`] of the binary
     157           80 :     pub fn symbols_version_requirement(&self) -> SymbolsVersionRequirement {
     158           80 :         SymbolsVersionRequirement::new(self.ptr.symbols_version_requirement())
     159           80 :     }
     160              : 
     161              :     /// Return an iterator over the  [`crate::elf::SymbolVersionDefinition`] of the binary
     162           80 :     pub fn symbols_version_definition(&self) -> SymbolsVersionDefinition {
     163           80 :         SymbolsVersionDefinition::new(self.ptr.symbols_version_definition())
     164           80 :     }
     165              : 
     166              :     /// Return an iterator over the  [`crate::elf::Notes`] of the binary
     167           80 :     pub fn notes(&self) -> ItNotes {
     168           80 :         ItNotes::new(self.ptr.notes())
     169           80 :     }
     170              : 
     171              :     /// Return an iterator over the `.plt.got` [`crate::elf::Relocation`] of the binary
     172           80 :     pub fn pltgot_relocations(&self) -> PltGotRelocations {
     173           80 :         PltGotRelocations::new(self.ptr.pltgot_relocations())
     174           80 :     }
     175              : 
     176              :     /// Return an iterator over the regular [`crate::elf::Relocation`] of the binary
     177           80 :     pub fn dynamic_relocations(&self) -> DynamicRelocations {
     178           80 :         DynamicRelocations::new(self.ptr.dynamic_relocations())
     179           80 :     }
     180              : 
     181              :     /// Return an iterator over the object-file (`.o`) [`crate::elf::Relocation`]
     182           80 :     pub fn object_relocations(&self) -> ObjectRelocations {
     183           80 :         ObjectRelocations::new(self.ptr.object_relocations())
     184           80 :     }
     185              : 
     186              :     /// Return an iterator over **all** [`crate::elf::Relocation`] of the binary
     187           80 :     pub fn relocations(&self) -> Relocations {
     188           80 :         Relocations::new(self.ptr.relocations())
     189           80 :     }
     190              : 
     191              :     /// Try to find the ELF section with the given name
     192           80 :     pub fn section_by_name(&self, name: &str) -> Option<Section> {
     193           80 :         into_optional(self.ptr.section_by_name(name))
     194           80 :     }
     195              : 
     196              :     /// Try to find the ELF relocation that takes place at the given address
     197           20 :     pub fn relocation_by_addr(&self, address: u64) -> Option<Relocation> {
     198           20 :         into_optional(self.ptr.relocation_by_addr(address))
     199           20 :     }
     200              : 
     201              :     /// Try to find the `.plt.got` relocation for the given symbol name
     202           20 :     pub fn relocation_for_symbol(&self, sym_name: &str) -> Option<Relocation> {
     203           20 :         into_optional(self.ptr.relocation_for_symbol(sym_name))
     204           20 :     }
     205              : 
     206              :     /// Try to find the symbol with the given name in the dynamic `.dynsym` table
     207           10 :     pub fn dynamic_symbol_by_name(&self, sym_name: &str) -> Option<Symbol> {
     208           10 :         into_optional(self.ptr.get_dynamic_symbol(sym_name))
     209           10 :     }
     210              : 
     211              :     /// Try to find the symbol with the given name in the debug `.symtab` table
     212           10 :     pub fn symtab_symbol_by_name(&self, sym_name: &str) -> Option<Symbol> {
     213           10 :         into_optional(self.ptr.get_symtab_symbol(sym_name))
     214           10 :     }
     215              : 
     216              :     /// Try to find the library (`DT_NEEDED`) with the given name
     217           20 :     pub fn get_library(&self, name: &str) -> Option<dynamic::Library> {
     218           20 :         into_optional(self.ptr.get_library(name))
     219           20 :     }
     220              : 
     221              :     /// Try to find the section that encompasses the given offset. `skip_nobits` can be used
     222              :     /// to include (or not) the `SHT_NOTBIT` sections
     223           20 :     pub fn section_from_offset(&self, offset: u64, skip_nobits: bool) -> Option<Section> {
     224           20 :         into_optional(self.ptr.section_from_offset(offset, skip_nobits))
     225           20 :     }
     226              : 
     227              :     /// Try to find the section that encompasses the given virtual address. `skip_nobits` can be used
     228              :     /// to include (or not) the `SHT_NOTBIT` sections
     229           20 :     pub fn section_from_virtual_address(&self, address: u64, skip_nobits: bool) -> Option<Section> {
     230           20 :         into_optional(self.ptr.section_from_virtual_address(address, skip_nobits))
     231           20 :     }
     232              : 
     233              :     /// Try to find the segment that encompasses the given virtual address
     234           20 :     pub fn segment_from_virtual_address(&self, address: u64) -> Option<Segment> {
     235           20 :         into_optional(self.ptr.segment_from_virtual_address(address))
     236           20 :     }
     237              : 
     238              :     /// Try to find the segment that encompasses the given offset
     239           20 :     pub fn segment_from_offset(&self, offset: u64) -> Option<Segment> {
     240           20 :         into_optional(self.ptr.segment_from_offset(offset))
     241           20 :     }
     242              : 
     243              :     /// Get a slice of the content at the given address.
     244           10 :     pub fn content_from_virtual_address(&self, address: u64, size: u64) -> &[u8] {
     245           10 :         to_slice!(self.ptr.get_content_from_virtual_address(address, size));
     246           10 :     }
     247              : 
     248              :     /// Convert the given virtual address into an offset
     249           20 :     pub fn virtual_address_to_offset(&self, address: u64) -> Result<u64, Error> {
     250           20 :         to_result!(ffi::ELF_Binary::virtual_address_to_offset, &self, address);
     251           20 :     }
     252              : 
     253              :     /// Return the array defined by the given tag (e.g.
     254              :     /// [`dynamic::Tag::INIT_ARRAY`]) with relocations applied (if any)
     255          240 :     pub fn get_relocated_dynamic_array(&self, tag: dynamic::Tag) -> Vec<u64> {
     256          240 :         Vec::from(
     257          240 :             self.ptr
     258          240 :                 .get_relocated_dynamic_array(u64::from(tag))
     259          240 :                 .as_slice(),
     260          240 :         )
     261          240 :     }
     262              : 
     263              :     /// True if the current binary is targeting Android
     264            0 :     pub fn is_targeting_android(&self) -> bool {
     265            0 :         self.ptr.is_targeting_android()
     266            0 :     }
     267              : 
     268              :     /// Get the integer value at the given virtual address
     269            4 :     pub fn get_int_from_virtual_address<T>(&self, addr: u64) -> Result<T, Error>
     270            4 :     where
     271            4 :         T: Num + cast::FromPrimitive + cast::ToPrimitive,
     272            4 :     {
     273            4 :         // Can't be in the generic trait because of:
     274            4 :         //   > for a trait to be "object safe" it needs to allow building a vtable to allow the call
     275            4 :         //   > to be resolvable dynamically; for more information visit
     276            4 :         //   > https://doc.rust-lang.org/reference/items/traits.html#object-safety
     277            4 :         if size_of::<T>() == size_of::<u8>() {
     278            1 :             to_conv_result!(
     279            1 :                 ffi::AbstractBinary::get_u8,
     280            1 :                 self.ptr.as_ref().unwrap().as_ref(),
     281            1 :                 |value| {
     282            1 :                     T::from_u8(value).expect(format!("Can't cast value: {}", value).as_str())
     283            1 :                 },
     284            1 :                 addr
     285              :             );
     286            3 :         }
     287            3 : 
     288            3 :         if size_of::<T>() == size_of::<u16>() {
     289            1 :             to_conv_result!(
     290            1 :                 ffi::AbstractBinary::get_u16,
     291            1 :                 self.ptr.as_ref().unwrap().as_ref(),
     292            1 :                 |value| {
     293            1 :                     T::from_u16(value).expect(format!("Can't cast value: {}", value).as_str())
     294            1 :                 },
     295            1 :                 addr
     296              :             );
     297            2 :         }
     298            2 : 
     299            2 :         if size_of::<T>() == size_of::<u32>() {
     300            1 :             to_conv_result!(
     301            1 :                 ffi::AbstractBinary::get_u32,
     302            1 :                 self.ptr.as_ref().unwrap().as_ref(),
     303            1 :                 |value| {
     304            1 :                     T::from_u32(value).expect(format!("Can't cast value: {}", value).as_str())
     305            1 :                 },
     306            1 :                 addr
     307              :             );
     308            1 :         }
     309            1 : 
     310            1 :         if size_of::<T>() == size_of::<u64>() {
     311            1 :             to_conv_result!(
     312            1 :                 ffi::AbstractBinary::get_u64,
     313            1 :                 self.ptr.as_ref().unwrap().as_ref(),
     314            1 :                 |value| {
     315            1 :                     T::from_u64(value).expect(format!("Can't cast value: {}", value).as_str())
     316            1 :                 },
     317            1 :                 addr
     318              :             );
     319            0 :         }
     320            0 : 
     321            0 :         Err(Error::NotSupported)
     322            4 :     }
     323              : 
     324              :     /// Write back the current ELF binary into the file specified in parameter
     325           10 :     pub fn write(&mut self, output: &Path) {
     326           10 :         self.ptr.as_mut().unwrap().write(output.to_str().unwrap());
     327           10 :     }
     328              : 
     329              :     /// Write back the current ELF binary into the file specified in parameter with the
     330              :     /// configuration provided in the second parameter.
     331           10 :     pub fn write_with_config(&mut self, output: &Path, config: Config) {
     332           10 :         self.ptr
     333           10 :             .as_mut()
     334           10 :             .unwrap()
     335           10 :             .write_with_config(output.to_str().unwrap(), config.to_ffi());
     336           10 :     }
     337              : 
     338              :     /// Add a library as dependency
     339           10 :     pub fn add_library<'a>(&'a mut self, library: &str) -> Library<'a> {
     340           10 :         Library::from_ffi(self.ptr.as_mut().unwrap().add_library(library))
     341           10 :     }
     342              : 
     343           80 :     pub fn functions(&self) -> generic::Functions {
     344           80 :         generic::Functions::new(self.ptr.functions())
     345           80 :     }
     346              : }
     347              : 
     348              : impl generic::Binary for Binary {
     349          400 :     fn as_generic(&self) -> &ffi::AbstractBinary {
     350          400 :         self.ptr.as_ref().unwrap().as_ref()
     351          400 :     }
     352              : 
     353            0 :     fn as_pin_mut_generic(&mut self) -> Pin<&mut ffi::AbstractBinary> {
     354            0 :         unsafe {
     355            0 :             Pin::new_unchecked({
     356            0 :                 (self.ptr.as_ref().unwrap().as_ref() as *const ffi::AbstractBinary
     357            0 :                     as *mut ffi::AbstractBinary)
     358            0 :                     .as_mut()
     359            0 :                     .unwrap()
     360            0 :             })
     361            0 :         }
     362            0 :     }
     363              : }
     364              : 
     365        15540 : declare_iterator!(
     366        15540 :     SymbolsVersion,
     367        15540 :     SymbolVersion<'a>,
     368        15540 :     ffi::ELF_SymbolVersion,
     369        15540 :     ffi::ELF_Binary,
     370        15540 :     ffi::ELF_Binary_it_symbols_version
     371        15540 : );
     372          150 : declare_iterator!(
     373          150 :     SymbolsVersionRequirement,
     374          150 :     SymbolVersionRequirement<'a>,
     375          150 :     ffi::ELF_SymbolVersionRequirement,
     376          150 :     ffi::ELF_Binary,
     377          150 :     ffi::ELF_Binary_it_symbols_version_requirement
     378          150 : );
     379           30 : declare_iterator!(
     380           30 :     SymbolsVersionDefinition,
     381           30 :     SymbolVersionDefinition<'a>,
     382           30 :     ffi::ELF_SymbolVersionDefinition,
     383           30 :     ffi::ELF_Binary,
     384           30 :     ffi::ELF_Binary_it_symbols_version_definition
     385           30 : );
        

Generated by: LCOV version 2.1-1