parse section names table header

main
pantonshire 1 year ago
parent 4d6521c5b4
commit d1ee14003c

@ -13,59 +13,79 @@ fn main() {
#[derive(Clone)]
struct InputBytes<'a> {
bytes: &'a [u8],
offset: usize,
}
impl<'a> InputBytes<'a> {
fn new(bytes: &'a [u8]) -> Self {
Self { bytes, offset: 0 }
Self { bytes }
}
fn new_offset(bytes: &'a [u8], offset: usize) -> Self {
Self::new(bytes).advanced(offset)
}
fn offset(&self) -> usize {
self.offset
fn prefix_len(&self, n: usize) -> usize {
self.bytes.len().min(n)
}
fn advance_offset(&self, n: usize) -> usize {
self.bytes.len().min(self.offset + n)
fn advanced(&self, n: usize) -> Self {
Self {
bytes: &self.bytes[self.prefix_len(n)..],
}
}
fn advance(&mut self, n: usize) -> &mut Self {
self.offset = self.advance_offset(n);
self.bytes = &self.bytes[self.prefix_len(n)..];
self
}
fn peek(&self, n: usize) -> Option<&'a [u8]> {
self.bytes.get(self.offset .. (self.offset + n))
fn peek(&self, n: usize) -> Result<&'a [u8], ParseError> {
self.bytes.get(..n).ok_or(ParseError::UnexpectedEof)
}
fn pop(&mut self, n: usize) -> Option<&'a [u8]> {
fn pop(&mut self, n: usize) -> Result<&'a [u8], ParseError> {
let res = self.peek(n);
self.advance(n);
if res.is_ok() {
self.advance(n);
}
res
}
fn pop_u16(&mut self, endianness: Endianness) -> Option<u16> {
let bytes = self.pop(2).and_then(|bytes| <[u8; 2]>::try_from(bytes).ok())?;
Some(endianness.read_u16(bytes))
fn pop_arr<const N: usize>(&mut self) -> Result<[u8; N], ParseError> {
self.pop(N).map(|bs| <[u8; N]>::try_from(bs).unwrap())
}
fn pop_u8(&mut self) -> Result<u8, ParseError> {
self.pop_arr::<1>().map(|[x]| x)
}
fn pop_u32(&mut self, endianness: Endianness) -> Option<u32> {
let bytes = self.pop(4).and_then(|bytes| <[u8; 4]>::try_from(bytes).ok())?;
Some(endianness.read_u32(bytes))
fn pop_u16(&mut self, endianness: Endianness) -> Result<u16, ParseError> {
let bytes = self.pop_arr::<2>()?;
Ok(endianness.read_u16(bytes))
}
fn pop_u64(&mut self, endianness: Endianness) -> Option<u64> {
let bytes = self.pop(8).and_then(|bytes| <[u8; 8]>::try_from(bytes).ok())?;
Some(endianness.read_u64(bytes))
fn pop_u32(&mut self, endianness: Endianness) -> Result<u32, ParseError> {
let bytes = self.pop_arr::<4>()?;
Ok(endianness.read_u32(bytes))
}
fn pop_word(&mut self, endianness: Endianness, width: Width) -> Option<u64> {
fn pop_u64(&mut self, endianness: Endianness) -> Result<u64, ParseError> {
let bytes = self.pop_arr::<8>()?;
Ok(endianness.read_u64(bytes))
}
fn pop_word(&mut self, endianness: Endianness, width: Width) -> Result<u64, ParseError> {
match width {
Width::Bits32 => self.pop_u32(endianness).map(u64::from),
Width::Bits64 => self.pop_u64(endianness),
}
}
fn pop_word_usize(&mut self, endianness: Endianness, width: Width) -> Result<usize, ParseError> {
self.pop_word(endianness, width)
.and_then(|x| usize::try_from(x).map_err(|_| ParseError::OffsetOutOfBounds))
}
fn hexdump(&self, n: usize) {
print!("{: <8}", "");
for i in 0..16 {
@ -91,9 +111,8 @@ impl<'a> InputBytes<'a> {
}
struct Elf<'a> {
bytes: &'a [u8],
width: Width,
endianness: Endianness,
elf_bytes: &'a [u8],
bl: ByteLayout,
header_version: u8,
abi: u8,
ty: ElfType,
@ -107,67 +126,68 @@ impl<'a> Elf<'a> {
hdr.hexdump(64);
let magic = hdr.pop(4).ok_or(ElfError)?;
let magic = hdr.pop(4)?;
if magic != &[0x7f, b'E', b'L', b'F'] {
return Err(ElfError);
return Err(ElfError::BadMagic);
}
let width = match hdr.pop(1) {
Some(&[1]) => Width::Bits32,
Some(&[2]) => Width::Bits64,
_ => return Err(ElfError),
};
let endianness = match hdr.pop(1) {
Some(&[1]) => Endianness::Little,
Some(&[2]) => Endianness::Big,
_ => return Err(ElfError),
let width = match hdr.pop_u8()? {
1 => Width::Bits32,
2 => Width::Bits64,
_ => return Err(ElfError::UnsupportedValue),
};
let Some(&[header_version]) = hdr.pop(1) else {
return Err(ElfError)
let endianness = match hdr.pop_u8()? {
1 => Endianness::Little,
2 => Endianness::Big,
_ => return Err(ElfError::UnsupportedValue),
};
let Some(&[abi]) = hdr.pop(1) else {
return Err(ElfError)
let bl = ByteLayout {
endianness,
width,
};
let header_version = hdr.pop_u8()?;
hdr.pop(8);
let abi = hdr.pop_u8()?;
let ty = match hdr.pop_u16(endianness) {
Some(1) => ElfType::Relocatable,
Some(2) => ElfType::Executable,
Some(3) => ElfType::Shared,
Some(4) => ElfType::Core,
_ => return Err(ElfError),
hdr.pop(8)?;
let ty = match hdr.pop_u16(endianness)? {
1 => ElfType::Relocatable,
2 => ElfType::Executable,
3 => ElfType::Shared,
4 => ElfType::Core,
_ => return Err(ElfError::UnsupportedValue),
};
let instruction_set = match hdr.pop_u16(endianness) {
Some(0) => None,
Some(0x02) => Some(InstructionSet::Sparc),
Some(0x03) => Some(InstructionSet::I386),
Some(0x08) => Some(InstructionSet::Mips),
Some(0x14) => Some(InstructionSet::PowerPc),
Some(0x28) => Some(InstructionSet::Arm),
Some(0x2a) => Some(InstructionSet::SuperH),
Some(0x32) => Some(InstructionSet::Ia64),
Some(0x3e) => Some(InstructionSet::Amd64),
Some(0xb7) => Some(InstructionSet::Aarch64),
Some(0xf3) => Some(InstructionSet::RiscV),
_ => return Err(ElfError),
let instruction_set = match hdr.pop_u16(endianness)? {
0 => None,
0x02 => Some(InstructionSet::Sparc),
0x03 => Some(InstructionSet::I386),
0x08 => Some(InstructionSet::Mips),
0x14 => Some(InstructionSet::PowerPc),
0x28 => Some(InstructionSet::Arm),
0x2a => Some(InstructionSet::SuperH),
0x32 => Some(InstructionSet::Ia64),
0x3e => Some(InstructionSet::Amd64),
0xb7 => Some(InstructionSet::Aarch64),
0xf3 => Some(InstructionSet::RiscV),
_ => return Err(ElfError::UnsupportedValue),
};
let elf_version = hdr.pop_u32(endianness).ok_or(ElfError)?;
let entry_offset = hdr.pop_word(endianness, width).ok_or(ElfError)?;
let pht_offset = hdr.pop_word(endianness, width).ok_or(ElfError)?;
let sht_offset = hdr.pop_word(endianness, width).ok_or(ElfError)?;
let flags = hdr.pop_u32(endianness).ok_or(ElfError)?;
let header_size = hdr.pop_u16(endianness).ok_or(ElfError)?;
let pht_entry_size = hdr.pop_u16(endianness).ok_or(ElfError)?;
let pht_len = hdr.pop_u16(endianness).ok_or(ElfError)?;
let sht_entry_size = hdr.pop_u16(endianness).ok_or(ElfError)?;
let sht_len = hdr.pop_u16(endianness).ok_or(ElfError)?;
let string_table_idx = hdr.pop_u16(endianness).ok_or(ElfError)?;
let elf_version = hdr.pop_u32(endianness)?;
let entry_offset = hdr.pop_word(endianness, width)?;
let pht_offset = hdr.pop_word_usize(endianness, width)?;
let sht_offset = hdr.pop_word_usize(endianness, width)?;
let flags = hdr.pop_u32(endianness)?;
let header_size = hdr.pop_u16(endianness)?;
let pht_entry_size = hdr.pop_u16(endianness)?;
let pht_len = hdr.pop_u16(endianness)?;
let sht_entry_size = hdr.pop_u16(endianness)?;
let sht_len = hdr.pop_u16(endianness)?;
let shstr_idx = hdr.pop_u16(endianness)?;
dbg!(entry_offset);
dbg!(pht_offset);
@ -178,14 +198,19 @@ impl<'a> Elf<'a> {
dbg!(pht_len);
dbg!(sht_entry_size);
dbg!(sht_len);
dbg!(string_table_idx);
dbg!(shstr_idx);
// TODO
let shstr = SectionHeader::parse(bl,
InputBytes::new_offset(bytes, sht_offset + usize::from(sht_entry_size * shstr_idx)))?;
let str_table = InputBytes::new_offset(bytes, shstr.offset);
str_table.hexdump(shstr.size);
Ok(Self {
bytes,
width,
endianness,
elf_bytes: bytes,
bl,
header_version,
abi,
ty,
@ -195,6 +220,54 @@ impl<'a> Elf<'a> {
}
}
struct SectionHeader {
name: u32,
// FIXME: parse type
ty: u32,
flags: u64,
addr: u64,
offset: usize,
size: usize,
link: u32,
info: u32,
align: u64,
ent_size: usize,
}
impl SectionHeader {
fn parse(bl: ByteLayout, mut bytes: InputBytes) -> Result<Self, ElfError> {
let name = bytes.pop_u32(bl.endianness)?;
let ty = bytes.pop_u32(bl.endianness)?;
let flags = bytes.pop_word(bl.endianness, bl.width)?;
let addr = bytes.pop_word(bl.endianness, bl.width)?;
let offset = bytes.pop_word_usize(bl.endianness, bl.width)?;
let size = bytes.pop_word_usize(bl.endianness, bl.width)?;
let link = bytes.pop_u32(bl.endianness)?;
let info = bytes.pop_u32(bl.endianness)?;
let align = bytes.pop_word(bl.endianness, bl.width)?;
let ent_size = bytes.pop_word_usize(bl.endianness, bl.width)?;
Ok(Self {
name,
ty,
flags,
addr,
offset,
size,
link,
info,
align,
ent_size,
})
}
}
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
struct ByteLayout {
endianness: Endianness,
width: Width,
}
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
enum Width {
Bits32,
@ -253,4 +326,24 @@ enum InstructionSet {
}
#[derive(Debug)]
struct ElfError;
enum ElfError {
BadMagic,
UnexpectedEof,
OffsetOutOfBounds,
UnsupportedValue,
}
impl From<ParseError> for ElfError {
fn from(value: ParseError) -> Self {
match value {
ParseError::UnexpectedEof => Self::UnexpectedEof,
ParseError::OffsetOutOfBounds => Self::OffsetOutOfBounds,
}
}
}
#[derive(Debug)]
enum ParseError {
UnexpectedEof,
OffsetOutOfBounds,
}

Loading…
Cancel
Save