use crate::prelude::*;
use crate::{
BinaryReader, BinaryReaderError, FromReader, Result, SectionLimited, Subsection, Subsections,
};
use core::ops::Range;
bitflags::bitflags! {
#[repr(transparent)]
#[derive(Debug, Clone, Copy, Default, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct SymbolFlags: u32 {
const BINDING_WEAK = 1 << 0;
const BINDING_LOCAL = 1 << 1;
const VISIBILITY_HIDDEN = 1 << 2;
const UNDEFINED = 1 << 4;
const EXPORTED = 1 << 5;
const EXPLICIT_NAME = 1 << 6;
const NO_STRIP = 1 << 7;
const TLS = 1 << 8;
const ABSOLUTE = 1 << 9;
}
#[repr(transparent)]
#[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Hash)]
pub struct SegmentFlags: u32 {
const STRINGS = 0x1;
const TLS = 0x2;
}
}
impl<'a> FromReader<'a> for SymbolFlags {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
Ok(Self::from_bits_retain(reader.read_var_u32()?))
}
}
impl<'a> FromReader<'a> for SegmentFlags {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
Ok(Self::from_bits_retain(reader.read_var_u32()?))
}
}
#[derive(Debug, Clone)]
pub struct LinkingSectionReader<'a> {
version: u32,
subsections: Subsections<'a, Linking<'a>>,
range: Range<usize>,
}
pub type SegmentMap<'a> = SectionLimited<'a, Segment<'a>>;
#[derive(Debug, Copy, Clone)]
pub struct Segment<'a> {
pub name: &'a str,
pub alignment: u32,
pub flags: SegmentFlags,
}
impl<'a> FromReader<'a> for Segment<'a> {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
let name = reader.read_unlimited_string()?;
let alignment = reader.read_var_u32()?;
let flags = reader.read()?;
Ok(Self {
name,
alignment,
flags,
})
}
}
pub type InitFuncMap<'a> = SectionLimited<'a, InitFunc>;
#[derive(Debug, Copy, Clone)]
pub struct InitFunc {
pub priority: u32,
pub symbol_index: u32,
}
impl<'a> FromReader<'a> for InitFunc {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
let priority = reader.read_var_u32()?;
let symbol_index = reader.read_var_u32()?;
Ok(Self {
priority,
symbol_index,
})
}
}
pub type ComdatMap<'a> = SectionLimited<'a, Comdat<'a>>;
#[derive(Debug, Clone)]
pub struct Comdat<'a> {
pub name: &'a str,
pub flags: u32,
pub symbols: SectionLimited<'a, ComdatSymbol>,
}
impl<'a> FromReader<'a> for Comdat<'a> {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
let name = reader.read_unlimited_string()?;
let flags = reader.read_var_u32()?;
let symbols = reader.skip(|reader| {
let count = reader.read_var_u32()?;
for _ in 0..count {
reader.read::<ComdatSymbol>()?;
}
Ok(())
})?;
Ok(Self {
name,
flags,
symbols: SectionLimited::new(symbols)?,
})
}
}
#[derive(Debug, Copy, Clone)]
pub struct ComdatSymbol {
pub kind: ComdatSymbolKind,
pub index: u32,
}
impl<'a> FromReader<'a> for ComdatSymbol {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
let kind = reader.read()?;
let index = reader.read_var_u32()?;
Ok(Self { kind, index })
}
}
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
pub enum ComdatSymbolKind {
Data,
Func,
Global,
Event,
Table,
Section,
}
impl<'a> FromReader<'a> for ComdatSymbolKind {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
let offset = reader.original_position();
match reader.read_u8()? {
0 => Ok(Self::Data),
1 => Ok(Self::Func),
2 => Ok(Self::Global),
3 => Ok(Self::Event),
4 => Ok(Self::Table),
5 => Ok(Self::Section),
k => Err(BinaryReader::invalid_leading_byte_error(
k,
"comdat symbol kind",
offset,
)),
}
}
}
pub type SymbolInfoMap<'a> = SectionLimited<'a, SymbolInfo<'a>>;
#[derive(Debug, Copy, Clone)]
pub enum SymbolInfo<'a> {
Func {
flags: SymbolFlags,
index: u32,
name: Option<&'a str>,
},
Data {
flags: SymbolFlags,
name: &'a str,
symbol: Option<DefinedDataSymbol>,
},
Global {
flags: SymbolFlags,
index: u32,
name: Option<&'a str>,
},
Section {
flags: SymbolFlags,
section: u32,
},
Event {
flags: SymbolFlags,
index: u32,
name: Option<&'a str>,
},
Table {
flags: SymbolFlags,
index: u32,
name: Option<&'a str>,
},
}
impl<'a> FromReader<'a> for SymbolInfo<'a> {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
let offset = reader.original_position();
let kind = reader.read_u8()?;
let flags: SymbolFlags = reader.read()?;
let defined = !flags.contains(SymbolFlags::UNDEFINED);
let explicit_name = flags.contains(SymbolFlags::EXPLICIT_NAME);
const SYMTAB_FUNCTION: u8 = 0;
const SYMTAB_DATA: u8 = 1;
const SYMTAB_GLOBAL: u8 = 2;
const SYMTAB_SECTION: u8 = 3;
const SYMTAB_EVENT: u8 = 4;
const SYMTAB_TABLE: u8 = 5;
match kind {
SYMTAB_FUNCTION | SYMTAB_GLOBAL | SYMTAB_EVENT | SYMTAB_TABLE => {
let index = reader.read_var_u32()?;
let name = match defined || explicit_name {
true => Some(reader.read_unlimited_string()?),
false => None,
};
Ok(match kind {
SYMTAB_FUNCTION => Self::Func { flags, index, name },
SYMTAB_GLOBAL => Self::Global { flags, index, name },
SYMTAB_EVENT => Self::Event { flags, index, name },
SYMTAB_TABLE => Self::Table { flags, index, name },
_ => unreachable!(),
})
}
SYMTAB_DATA => {
let name = reader.read_unlimited_string()?;
let data = match defined {
true => Some(reader.read()?),
false => None,
};
Ok(Self::Data {
flags,
name,
symbol: data,
})
}
SYMTAB_SECTION => {
let section = reader.read_var_u32()?;
Ok(Self::Section { flags, section })
}
k => Err(BinaryReader::invalid_leading_byte_error(
k,
"symbol kind",
offset,
)),
}
}
}
#[derive(Debug, Copy, Clone)]
pub struct DefinedDataSymbol {
pub index: u32,
pub offset: u32,
pub size: u32,
}
impl<'a> FromReader<'a> for DefinedDataSymbol {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
let index = reader.read_var_u32()?;
let offset = reader.read_var_u32()?;
let size = reader.read_var_u32()?;
Ok(Self {
index,
offset,
size,
})
}
}
#[derive(Debug, Clone)]
pub enum Linking<'a> {
SegmentInfo(SegmentMap<'a>),
InitFuncs(InitFuncMap<'a>),
ComdatInfo(ComdatMap<'a>),
SymbolTable(SymbolInfoMap<'a>),
Unknown {
ty: u8,
data: &'a [u8],
range: Range<usize>,
},
}
impl<'a> Subsection<'a> for Linking<'a> {
fn from_reader(id: u8, reader: BinaryReader<'a>) -> Result<Self> {
let data = reader.remaining_buffer();
let offset = reader.original_position();
Ok(match id {
5 => Self::SegmentInfo(SegmentMap::new(reader)?),
6 => Self::InitFuncs(InitFuncMap::new(reader)?),
7 => Self::ComdatInfo(ComdatMap::new(reader)?),
8 => Self::SymbolTable(SymbolInfoMap::new(reader)?),
ty => Self::Unknown {
ty,
data,
range: offset..offset + data.len(),
},
})
}
}
impl<'a> LinkingSectionReader<'a> {
pub fn new(mut reader: BinaryReader<'a>) -> Result<Self> {
let range = reader.range();
let offset = reader.original_position();
let version = reader.read_var_u32()?;
if version != 2 {
return Err(BinaryReaderError::new(
format!("unsupported linking section version: {version}"),
offset,
));
}
let subsections = Subsections::new(reader.shrink());
Ok(Self {
version,
subsections,
range,
})
}
pub fn version(&self) -> u32 {
self.version
}
pub fn original_position(&self) -> usize {
self.subsections.original_position()
}
pub fn range(&self) -> Range<usize> {
self.range.clone()
}
pub fn subsections(&self) -> Subsections<'a, Linking<'a>> {
self.subsections.clone()
}
}
impl<'a> IntoIterator for LinkingSectionReader<'a> {
type Item = Result<Linking<'a>>;
type IntoIter = Subsections<'a, Linking<'a>>;
fn into_iter(self) -> Self::IntoIter {
self.subsections
}
}