1use {
2 crate::errors::DisassemblerError,
3 object::{Endianness, read::elf::ElfFile64},
4 serde::{Deserialize, Serialize},
5 std::fmt::Debug,
6};
7
8pub const PF_X: u8 = 0x01;
10pub const PF_W: u8 = 0x02;
11pub const PF_R: u8 = 0x04;
12
13#[allow(non_camel_case_types)]
14#[derive(Debug, Clone, Serialize, Deserialize)]
15#[repr(u32)]
16pub enum ProgramType {
17 PT_NULL = 0x00, PT_LOAD = 0x01, PT_DYNAMIC = 0x02, PT_INTERP = 0x03, PT_NOTE = 0x04, PT_SHLIB = 0x05, PT_PHDR = 0x06, PT_TLS = 0x07, }
26
27impl TryFrom<u32> for ProgramType {
28 type Error = DisassemblerError;
29
30 fn try_from(value: u32) -> Result<Self, Self::Error> {
31 Ok(match value {
32 0 => Self::PT_NULL,
33 1 => Self::PT_LOAD,
34 2 => Self::PT_DYNAMIC,
35 3 => Self::PT_INTERP,
36 4 => Self::PT_NOTE,
37 5 => Self::PT_SHLIB,
38 6 => Self::PT_PHDR,
39 7 => Self::PT_TLS,
40 _ => return Err(DisassemblerError::InvalidProgramType),
41 })
42 }
43}
44
45impl From<ProgramType> for u32 {
46 fn from(val: ProgramType) -> Self {
47 match val {
48 ProgramType::PT_NULL => 0,
49 ProgramType::PT_LOAD => 1,
50 ProgramType::PT_DYNAMIC => 2,
51 ProgramType::PT_INTERP => 3,
52 ProgramType::PT_NOTE => 4,
53 ProgramType::PT_SHLIB => 5,
54 ProgramType::PT_PHDR => 6,
55 ProgramType::PT_TLS => 7,
56 }
57 }
58}
59
60impl From<ProgramType> for &str {
61 fn from(val: ProgramType) -> Self {
62 match val {
63 ProgramType::PT_NULL => "PT_NULL",
64 ProgramType::PT_LOAD => "PT_LOAD",
65 ProgramType::PT_DYNAMIC => "PT_DYNAMIC",
66 ProgramType::PT_INTERP => "PT_INTERP",
67 ProgramType::PT_NOTE => "PT_NOTE",
68 ProgramType::PT_SHLIB => "PT_SHLIB",
69 ProgramType::PT_PHDR => "PT_PHDR",
70 ProgramType::PT_TLS => "PT_TLS",
71 }
72 }
73}
74
75#[derive(Debug, Clone, Serialize, Deserialize)]
76pub struct ProgramFlags(pub u32);
77
78impl From<u32> for ProgramFlags {
79 fn from(value: u32) -> Self {
80 Self(value & 7)
81 }
82}
83
84impl std::fmt::Display for ProgramFlags {
85 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
86 let x = match self.0 & PF_X as u32 == PF_X as u32 {
87 true => "X",
88 false => "*",
89 };
90
91 let r = match self.0 & PF_R as u32 == PF_R as u32 {
92 true => "R",
93 false => "*",
94 };
95
96 let w = match self.0 & PF_W as u32 == PF_W as u32 {
97 true => "W",
98 false => "*",
99 };
100 f.write_str(&format!("{}/{}/{}", r, w, x))
101 }
102}
103
104#[derive(Debug, Clone, Serialize, Deserialize)]
105pub struct ProgramHeader {
106 pub p_type: ProgramType, pub p_flags: ProgramFlags, pub p_offset: u64, pub p_vaddr: u64, pub p_paddr: u64, pub p_filesz: u64, pub p_memsz: u64, pub p_align: u64, }
115
116impl ProgramHeader {
117 pub fn from_elf_file(elf_file: &ElfFile64<Endianness>) -> Result<Vec<Self>, DisassemblerError> {
118 let endian = elf_file.endian();
119 let program_headers_data = elf_file.elf_program_headers();
120
121 let mut program_headers = Vec::new();
122 for ph in program_headers_data {
123 let p_type = ProgramType::try_from(ph.p_type.get(endian))?;
124 let p_flags = ProgramFlags::from(ph.p_flags.get(endian));
125 let p_offset = ph.p_offset.get(endian);
126 let p_vaddr = ph.p_vaddr.get(endian);
127 let p_paddr = ph.p_paddr.get(endian);
128 let p_filesz = ph.p_filesz.get(endian);
129 let p_memsz = ph.p_memsz.get(endian);
130 let p_align = ph.p_align.get(endian);
131
132 program_headers.push(ProgramHeader {
133 p_type,
134 p_flags,
135 p_offset,
136 p_vaddr,
137 p_paddr,
138 p_filesz,
139 p_memsz,
140 p_align,
141 });
142 }
143
144 Ok(program_headers)
145 }
146
147 pub fn to_bytes(&self) -> Vec<u8> {
148 let mut b = (self.p_type.clone() as u32).to_le_bytes().to_vec();
149 b.extend_from_slice(&self.p_flags.0.to_le_bytes());
150 b.extend_from_slice(&self.p_offset.to_le_bytes());
151 b.extend_from_slice(&self.p_vaddr.to_le_bytes());
152 b.extend_from_slice(&self.p_paddr.to_le_bytes());
153 b.extend_from_slice(&self.p_filesz.to_le_bytes());
154 b.extend_from_slice(&self.p_memsz.to_le_bytes());
155 b.extend_from_slice(&self.p_align.to_le_bytes());
156 b
157 }
158}
159
160#[cfg(test)]
161mod tests {
162 use {super::*, crate::program::Program, hex_literal::hex};
163
164 #[test]
165 fn test_program_headers() {
166 let original_bytes = hex!(
167 "7F454C460201010000000000000000000300F700010000002001000000000000400000000000000028020000000000000000000040003800030040000600050001000000050000002001000000000000200100000000000020010000000000003000000000000000300000000000000000100000000000000100000004000000C001000000000000C001000000000000C0010000000000003C000000000000003C000000000000000010000000000000020000000600000050010000000000005001000000000000500100000000000070000000000000007000000000000000080000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000007912A000000000007911182900000000B7000000010000002D21010000000000B70000000000000095000000000000001E0000000000000004000000000000000600000000000000C0010000000000000B0000000000000018000000000000000500000000000000F0010000000000000A000000000000000C00000000000000160000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001000000120001002001000000000000300000000000000000656E747279706F696E7400002E74657874002E64796E737472002E64796E73796D002E64796E616D6963002E73687374727461620000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001000000010000000600000000000000200100000000000020010000000000003000000000000000000000000000000008000000000000000000000000000000170000000600000003000000000000005001000000000000500100000000000070000000000000000400000000000000080000000000000010000000000000000F0000000B0000000200000000000000C001000000000000C001000000000000300000000000000004000000010000000800000000000000180000000000000007000000030000000200000000000000F001000000000000F0010000000000000C00000000000000000000000000000001000000000000000000000000000000200000000300000000000000000000000000000000000000FC010000000000002A00000000000000000000000000000001000000000000000000000000000000"
168 );
169 let program = Program::from_bytes(&original_bytes).unwrap();
170
171 assert_eq!(program.program_headers.len(), 3);
173
174 for (i, program_header) in program.program_headers.iter().enumerate() {
176 let serialized = program_header.to_bytes();
177 let header_offset = 0x40 + (i * 56);
178 let original_header_bytes = &original_bytes[header_offset..header_offset + 56];
179 assert_eq!(serialized, original_header_bytes,);
180 }
181 }
182
183 #[test]
184 fn test_program_type_conversions() {
185 assert!(matches!(ProgramType::try_from(0), Ok(ProgramType::PT_NULL)));
187 assert!(matches!(ProgramType::try_from(1), Ok(ProgramType::PT_LOAD)));
188 assert!(matches!(
189 ProgramType::try_from(2),
190 Ok(ProgramType::PT_DYNAMIC)
191 ));
192 assert!(matches!(
193 ProgramType::try_from(3),
194 Ok(ProgramType::PT_INTERP)
195 ));
196 assert!(matches!(ProgramType::try_from(4), Ok(ProgramType::PT_NOTE)));
197 assert!(matches!(
198 ProgramType::try_from(5),
199 Ok(ProgramType::PT_SHLIB)
200 ));
201 assert!(matches!(ProgramType::try_from(6), Ok(ProgramType::PT_PHDR)));
202 assert!(matches!(ProgramType::try_from(7), Ok(ProgramType::PT_TLS)));
203
204 assert!(ProgramType::try_from(99).is_err());
206
207 assert_eq!(u32::from(ProgramType::PT_NULL), 0);
209 assert_eq!(u32::from(ProgramType::PT_LOAD), 1);
210 assert_eq!(u32::from(ProgramType::PT_DYNAMIC), 2);
211 assert_eq!(u32::from(ProgramType::PT_INTERP), 3);
212 assert_eq!(u32::from(ProgramType::PT_NOTE), 4);
213 assert_eq!(u32::from(ProgramType::PT_SHLIB), 5);
214 assert_eq!(u32::from(ProgramType::PT_PHDR), 6);
215 assert_eq!(u32::from(ProgramType::PT_TLS), 7);
216
217 assert_eq!(<&str>::from(ProgramType::PT_NULL), "PT_NULL");
219 assert_eq!(<&str>::from(ProgramType::PT_LOAD), "PT_LOAD");
220 assert_eq!(<&str>::from(ProgramType::PT_DYNAMIC), "PT_DYNAMIC");
221 assert_eq!(<&str>::from(ProgramType::PT_INTERP), "PT_INTERP");
222 assert_eq!(<&str>::from(ProgramType::PT_NOTE), "PT_NOTE");
223 assert_eq!(<&str>::from(ProgramType::PT_SHLIB), "PT_SHLIB");
224 assert_eq!(<&str>::from(ProgramType::PT_PHDR), "PT_PHDR");
225 assert_eq!(<&str>::from(ProgramType::PT_TLS), "PT_TLS");
226 }
227
228 #[test]
229 fn test_program_flags_display() {
230 assert_eq!(ProgramFlags::from(0).to_string(), "*/*/*"); assert_eq!(ProgramFlags::from(1).to_string(), "*/*/X"); assert_eq!(ProgramFlags::from(2).to_string(), "*/W/*"); assert_eq!(ProgramFlags::from(4).to_string(), "R/*/*"); assert_eq!(ProgramFlags::from(5).to_string(), "R/*/X"); assert_eq!(ProgramFlags::from(6).to_string(), "R/W/*"); assert_eq!(ProgramFlags::from(7).to_string(), "R/W/X"); }
239}